Eleventy has some pretty cool post dating features. For my blog
though, I added a draft feature which led me to add my own custom date property,
datePublished
. This also let me avoid subtle bugs that could come up when the
filesystem or git database changed for reasons other than a post edit.
This system worked pretty well but there were a few drawbacks:
- I had to add the
datePublished
to each post's frontmatter - Posts appear out of order in the file system
To solve these problems I configured 11ty to derive the post's datePublished
from an ISO date at the beginning of the file. One of the advantages of ISO-8601
date strings, and thus a reason why they are the only legit date format,
is that they are lexically sortable. That's a fancy way to say that if you sort
them character-by-character, you'll always end up sorting them by date.
So my solution involved parsing either the input path when computing the
datePublished
and the permalink
using a regular expression:
const POST_DATE_RE = /(?<prefix>^.*\/)(?<date>\d{4}-(?:[0]\d|1[0-2])-(?:[0-2]\d|3[01]))-(?<suffix>.+)/;
Regex with three named capture groups:
prefix
is anything followed by a/
date
is an ISO-8601 date string- a
-
suffix
is anything after the-
With this regex, I laid out my posts dir like so:
data:image/s3,"s3://crabby-images/cec5d/cec5d571c8fdf956698ac05a72ae5d692123dff3" alt=""
posts ├── 2022-01-07-lets-write-a-redux-controller-for-web-components.md ├── 2022-11-14-form-associated-custom-elements.md ├── 2022-12-12-micro-dreidle.md ├── 2022-12-25-8-days-5783.md ├── 2023-01-01-microbit-countdown.md ├── 2023-01-15-11ty-svg-sprites.md ├── 2023-01-31-cheap-netlify-11ty-rebuilds.md ├── 2023-02-18-splitjoin-nvim.md ├── 2023-02-19-microcopy-reactive-controller.md ├── 2023-03-19-microbit-spruce-up-tug-of-led.md ├── 2023-04-11-webc-impressions.md ├── 2023-04-13-11ty-wrap-emoji.md ├── 2023-04-14-eli5-web-components.md ├── 2023-04-14-sefira-isru-hag-pesah.md ├── 2023-04-23-webc-nvim.md ├── 2023-04-26-adelman.md ├── 2023-05-09-import-map-cdn.md ├── 2023-05-21-markdown-images-treesitter.md ├── 2023-07-10-debugging-gnome-extensions-dbus-run-session.md ├── 2023-07-23-webc-dsd-slot-workaround.md ├── 2023-07-28-sort-and-date-11ty-posts-by-name.md ├── index.11tydata.cjs ├── index.css ├── index.webc ├── lets-build-web-components │ ├── part-1-the-standards.md │ ├── part-2-the-polyfills.md │ ├── part-3-vanilla-components.md │ ├── part-4-polymer-library.md │ ├── part-5-litelement.md │ ├── part-6-gluon.md │ ├── part-7-hybrids.md │ └── part-8-mythbusters.md ├── posts.11tydata.cjs └── posts.yaml2 directories, 45 files
Next step is to extract the datePublished
from the filename, and rewrite the
permalink
to remove the date.
module.exports = {
eleventyComputed: {
datePublished({ datePublished, page }) {
const { date } = page.inputPath?.match(POST_DATE_RE)?.groups ?? {};
if (!datePublished && date)
return new Date(date);
else
return datePublished;
},
permalink({ permalink, page }) {
const match = page.inputPath.match(POST_DATE_RE);
if (match && !page.filePathStem.endsWith('/index'))
return `${page.filePathStem}/index.html`
else
return permalink
}
}
}
As a bonus, I wrote this sorting function for neo-tree to sort only my posts directory in descending order:
require'neo-tree'.setup {
-- ...
sort_function = function(a, b)
if a.path:match[[bennypowers.dev/posts/.+]] and a.type == b.type then
return a.path > b.path
-- default sort
elseif a.type == b.type then
return a.path < b.path
else
return a.type < b.type
end
end,
-- ...
}
Hope you found this helpful.