Compare commits
No commits in common. "54ec74c6edcca6f84c96502ebbfa4bb6167fe116" and "a0dc4211893a131ac05d90b76b6268d67ac156cb" have entirely different histories.
54ec74c6ed
...
a0dc421189
54
README.md
54
README.md
@ -8,62 +8,10 @@ My blog, based on the very helpful eleventy-base-blog v8.
|
|||||||
### Metadata
|
### Metadata
|
||||||
Site metadata such as author info, title, etc. lives in _data/metadata.js. Links on the /me page, and default post images are also configured here.
|
Site metadata such as author info, title, etc. lives in _data/metadata.js. Links on the /me page, and default post images are also configured here.
|
||||||
|
|
||||||
#### /me
|
### How to add a cowsay to a post with the custom nunjucks filter
|
||||||
Links at /me are configured in `metadata.js` like so:
|
|
||||||
``` javascript
|
|
||||||
socialLinks: [
|
|
||||||
{
|
|
||||||
title: "My Blog",
|
|
||||||
linkURL: "https://nathanupchurch.com",
|
|
||||||
linkDisplay: "My Blog",
|
|
||||||
iconURL: "/img/logo.svg"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
```
|
|
||||||
Any links added here will render on the /me page.
|
|
||||||
|
|
||||||
#### Webrings
|
|
||||||
Webrings configured in `metadata.js` will display in the footer of most pages beneath the copyright notice:
|
|
||||||
``` javascript
|
|
||||||
webrings: [
|
|
||||||
{
|
|
||||||
name: "Fediring",
|
|
||||||
ringURL: "https://fediring.net/",
|
|
||||||
previousURL: "https://fediring.net/previous?host=nathanupchurch.com",
|
|
||||||
nextURL: "https://fediring.net/next?host=nathanupchurch.com"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
```
|
|
||||||
Any links added here will render on the /me page.
|
|
||||||
|
|
||||||
### Add a cowsay to a post
|
|
||||||
You can use the `cowsay` filter to output a captioned `<figure>` containing a copy of an output from the cowsay program. Instead of using the usual three backticks, this method is accessible to visually impaired users thanks to the automatic captioning. Here's how to do it:
|
You can use the `cowsay` filter to output a captioned `<figure>` containing a copy of an output from the cowsay program. Instead of using the usual three backticks, this method is accessible to visually impaired users thanks to the automatic captioning. Here's how to do it:
|
||||||
|
|
||||||
1. Add a copy of the cowsay output you'd like to display to _data/cowList.js. Be sure and escape any backslashes.
|
1. Add a copy of the cowsay output you'd like to display to _data/cowList.js. Be sure and escape any backslashes.
|
||||||
2. Use the filter like this: `{{ cowList.name | cowsay | safe }}`.
|
2. Use the filter like this: `{{ cowList.name | cowsay | safe }}`.
|
||||||
|
|
||||||
The `safe` filter is necessary so that Eleventy doesn't sanitize our HTML.
|
The `safe` filter is necessary so that Eleventy doesn't sanitize our HTML.
|
||||||
|
|
||||||
### Embed a toot in a post
|
|
||||||
Embed a toot from Mastodon using the `toot` shortcode:
|
|
||||||
```
|
|
||||||
{% toot "instance", "tootID" %}
|
|
||||||
```
|
|
||||||
For example:
|
|
||||||
```
|
|
||||||
{% toot "lounge.town", "112672230453089386" %}
|
|
||||||
```
|
|
||||||
Embedded toots are live, so if the toot or instance is not reachable, the toot will not be shown on the page. I haven't built any error handling for this yet.
|
|
||||||
|
|
||||||
### Adding comments via Mastodon
|
|
||||||
Simply add a toot ID to a post's frontmatter using the `mastodon_id` key in order to enable commenting via Mastodon:
|
|
||||||
|
|
||||||
```
|
|
||||||
mastodon_id: "111688829907363670"
|
|
||||||
```
|
|
||||||
|
|
||||||
The toot at the given ID will be embedded in a comment section at the bottom of the post; users can reply to the post on Mastodon and their replies will automatically be embedded on the post.
|
|
||||||
|
|
||||||
Like embedding a toot, comments are live and rely on the instance and toots / comments to be reachable on the remote server.
|
|
||||||
|
|
||||||
I also have not added any error handling for this, nor have I added any sort of moderation at this point.
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export default {
|
module.exports = {
|
||||||
onScience: `
|
onScience: `
|
||||||
_________________________________________
|
_________________________________________
|
||||||
( Once, when the secrets of science were )
|
( Once, when the secrets of science were )
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export default {
|
module.exports = {
|
||||||
title: "Nathan Upchurch",
|
title: "Nathan Upchurch",
|
||||||
logo: "/img/logo.svg",
|
logo: "/img/logo.svg",
|
||||||
url: "https://nathanupchurch.com/",
|
url: "https://nathanupchurch.com/",
|
||||||
|
@ -25,8 +25,27 @@
|
|||||||
</wc-comment>
|
</wc-comment>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script type="module">
|
<script>
|
||||||
import {dateSuffixAdder, monthMap, timeFormatter} from "../../js/modules/mastodonDateTools.js";
|
const monthMap = ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"];
|
||||||
|
|
||||||
|
const dateSuffixAdder = (date) => {
|
||||||
|
if (date > 9 && date < 20) {
|
||||||
|
return "th";
|
||||||
|
} else {
|
||||||
|
let dateString = date < 10 ? "0" + date : "" + date;
|
||||||
|
if (dateString[1] < 4 && dateString[1] > 0) {
|
||||||
|
return dateString[1] == 1 ? "st" :
|
||||||
|
dateString[1] == 2 ? "nd" :
|
||||||
|
dateString[1] == 3 ? "rd" : null;
|
||||||
|
} else {
|
||||||
|
return "th"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const timeFormatter = (hours, minutes) => {
|
||||||
|
return `${hours < 12 ? hours : hours - 12}:${minutes < 10 ? "0" : ""}${minutes} ${hours < 12 ? "AM" : "PM"}`
|
||||||
|
}
|
||||||
|
|
||||||
const renderComment = (comment, target, parentIdm) => {
|
const renderComment = (comment, target, parentIdm) => {
|
||||||
const node = document
|
const node = document
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export default {
|
module.exports = {
|
||||||
tags: [
|
tags: [
|
||||||
"posts"
|
"posts"
|
||||||
],
|
],
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
export default {
|
module.exports = {
|
||||||
eleventyExcludeFromCollections: true
|
eleventyExcludeFromCollections: true
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export default {
|
module.exports = {
|
||||||
tags: [
|
tags: [
|
||||||
"now"
|
"now"
|
||||||
],
|
],
|
||||||
|
@ -1,21 +1,20 @@
|
|||||||
import { DateTime } from "luxon";
|
const { DateTime } = require("luxon");
|
||||||
import markdownIt from "markdown-it";
|
const markdownIt = require("markdown-it");
|
||||||
import markdownItFootnote from "markdown-it-footnote";
|
const markdownItFootnote = require("markdown-it-footnote");
|
||||||
import markdownItAnchor from "markdown-it-anchor";
|
const markdownItAnchor = require("markdown-it-anchor");
|
||||||
import mdfigcaption from 'markdown-it-image-figures';
|
const mdfigcaption = require('markdown-it-image-figures');
|
||||||
import pluginRss from "@11ty/eleventy-plugin-rss";
|
const pluginRss = require("@11ty/eleventy-plugin-rss");
|
||||||
import pluginSyntaxHighlight from "@11ty/eleventy-plugin-syntaxhighlight";
|
const pluginSyntaxHighlight = require("@11ty/eleventy-plugin-syntaxhighlight");
|
||||||
import pluginBundle from "@11ty/eleventy-plugin-bundle";
|
const pluginBundle = require("@11ty/eleventy-plugin-bundle");
|
||||||
import pluginNavigation from "@11ty/eleventy-navigation";
|
const pluginNavigation = require("@11ty/eleventy-navigation");
|
||||||
import prettier from "prettier";
|
const prettier = require("prettier");
|
||||||
import { EleventyHtmlBasePlugin } from "@11ty/eleventy";
|
const { EleventyHtmlBasePlugin } = require("@11ty/eleventy");
|
||||||
import {dateSuffixAdder, monthMap, timeFormatter} from "./public/js/modules/mastodonDateTools.js";
|
|
||||||
|
|
||||||
const figoptions = {
|
const figoptions = {
|
||||||
figcaption: true
|
figcaption: true
|
||||||
};
|
};
|
||||||
|
|
||||||
export default async function(eleventyConfig) {
|
module.exports = eleventyConfig => {
|
||||||
|
|
||||||
// Helper Functions
|
// Helper Functions
|
||||||
const multiReplace = (text, replacementTable) => {
|
const multiReplace = (text, replacementTable) => {
|
||||||
@ -89,38 +88,12 @@ export default async function(eleventyConfig) {
|
|||||||
`;
|
`;
|
||||||
});
|
});
|
||||||
|
|
||||||
eleventyConfig.addAsyncShortcode("toot", async function(instance, ID) {
|
|
||||||
const tootData = await fetch(
|
|
||||||
`https://${instance}/api/v1/statuses/${ID}`
|
|
||||||
);
|
|
||||||
|
|
||||||
const toot = await tootData.json();
|
|
||||||
|
|
||||||
const dateObj = new Date(toot.created_at);
|
|
||||||
|
|
||||||
const dateTime = `${dateObj.getDate()}${dateSuffixAdder(dateObj.getDate())} of ${monthMap[dateObj.getMonth()]}, ${dateObj.getFullYear()}, at ${timeFormatter(dateObj.getHours(), dateObj.getMinutes())}`;
|
|
||||||
|
|
||||||
return `
|
|
||||||
<wc-toot
|
|
||||||
author_name="${toot.account.display_name}"
|
|
||||||
author_url="${toot.url.replace(/\/[0-9]+/, "")}"
|
|
||||||
author_username="${toot.account.username}"
|
|
||||||
avatar_url="${toot.account.avatar_static}"
|
|
||||||
toot_content="${toot.content}"
|
|
||||||
toot_url="${toot.url}"
|
|
||||||
publish_date="${dateTime}"
|
|
||||||
sharp_corner="">
|
|
||||||
</wc-toot>
|
|
||||||
`
|
|
||||||
});
|
|
||||||
|
|
||||||
// Passthrough
|
// Passthrough
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/xsl/*': "/xsl/" });
|
eleventyConfig.addPassthroughCopy({ 'public/xsl/*': "/xsl/" });
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/img/*': "/img/" });
|
eleventyConfig.addPassthroughCopy({ 'public/img/*': "/img/" });
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/robots.txt': "/" });
|
eleventyConfig.addPassthroughCopy({ 'public/robots.txt': "/" });
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/js/*': "/js/" });
|
eleventyConfig.addPassthroughCopy({ 'public/js/*': "/js/" });
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/js/webComponents/*': "/js/webComponents" });
|
eleventyConfig.addPassthroughCopy({ 'public/js/webComponents/*': "/js/webComponents" });
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/js/modules/*': "/js/modules" });
|
|
||||||
// Copying so that basic.xsl can use it
|
// Copying so that basic.xsl can use it
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/css/index.css': "/css/index.css" });
|
eleventyConfig.addPassthroughCopy({ 'public/css/index.css': "/css/index.css" });
|
||||||
eleventyConfig.addPassthroughCopy({ 'public/css/webfonts/*': "/css/webfonts/" });
|
eleventyConfig.addPassthroughCopy({ 'public/css/webfonts/*': "/css/webfonts/" });
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
"name": "nathanupchurch.com",
|
"name": "nathanupchurch.com",
|
||||||
"version": "8.0.0",
|
"version": "8.0.0",
|
||||||
"description": "The personal website and blog of Nathan Upchurch",
|
"description": "The personal website and blog of Nathan Upchurch",
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "npx @11ty/eleventy",
|
"build": "npx @11ty/eleventy",
|
||||||
"build-ghpages": "npx @11ty/eleventy --pathprefix=/eleventy-base-blog/",
|
"build-ghpages": "npx @11ty/eleventy --pathprefix=/eleventy-base-blog/",
|
||||||
@ -29,7 +28,7 @@
|
|||||||
},
|
},
|
||||||
"homepage": "https://upchur.ch/gitea/n_u/nathanupchurch.com",
|
"homepage": "https://upchur.ch/gitea/n_u/nathanupchurch.com",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@11ty/eleventy": "3.0.0-alpha.13",
|
"@11ty/eleventy": "^2.0.1",
|
||||||
"@11ty/eleventy-img": "^3.1.0",
|
"@11ty/eleventy-img": "^3.1.0",
|
||||||
"@11ty/eleventy-navigation": "^0.3.5",
|
"@11ty/eleventy-navigation": "^0.3.5",
|
||||||
"@11ty/eleventy-plugin-bundle": "^1.0.4",
|
"@11ty/eleventy-plugin-bundle": "^1.0.4",
|
||||||
|
@ -18,10 +18,9 @@
|
|||||||
--font-variation-default: "opsz" 18, "wght" 310, "SOFT" 40, "WONK" 0;
|
--font-variation-default: "opsz" 18, "wght" 310, "SOFT" 40, "WONK" 0;
|
||||||
--font-variation-ui: "wght" 500;
|
--font-variation-ui: "wght" 500;
|
||||||
--font-family-monospace: Consolas, Menlo, Monaco, Andale Mono WT, Andale Mono, Lucida Console, Lucida Sans Typewriter, DejaVu Sans Mono, Bitstream Vera Sans Mono, Liberation Mono, Nimbus Mono L, Courier New, Courier, monospace;
|
--font-family-monospace: Consolas, Menlo, Monaco, Andale Mono WT, Andale Mono, Lucida Console, Lucida Sans Typewriter, DejaVu Sans Mono, Bitstream Vera Sans Mono, Liberation Mono, Nimbus Mono L, Courier New, Courier, monospace;
|
||||||
--meta-font-family: var(--font-family-ui);
|
--meta-font-size: var(--step--1);
|
||||||
--meta-font-size: var(--step--2);
|
--meta-font-style: italic;
|
||||||
--meta-font-style: normal;
|
--meta-font-variation-settings: "opsz" 12, "wght" 310, "SOFT" 40, "WONK" 0;
|
||||||
--meta-font-variation-settings: var(--font-variation-ui);
|
|
||||||
--ui-letter-spacing: calc(var(--space-3xs) * 0.5);
|
--ui-letter-spacing: calc(var(--space-3xs) * 0.5);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -387,30 +386,20 @@ table th {
|
|||||||
.continue-discussion button {
|
.continue-discussion button {
|
||||||
margin-top: var(--space-xs);
|
margin-top: var(--space-xs);
|
||||||
}
|
}
|
||||||
wc-comment::part(author-link), wc-toot::part(author-link) {
|
wc-comment::part(author-link) {
|
||||||
font-size: var(--step-0);
|
font-size: var(--step-0);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
wc-comment::part(main) {
|
wc-comment::part(main) {
|
||||||
margin-bottom: var(--space-l);
|
margin-bottom: var(--space-l);
|
||||||
}
|
}
|
||||||
wc-comment::part(publish-date), wc-toot::part(publish-date) {
|
wc-comment::part(publish-date) {
|
||||||
font-family: var(--meta-font-family);
|
|
||||||
font-size: var(--meta-font-size);
|
font-size: var(--meta-font-size);
|
||||||
font-style: var(--meta-font-style);
|
font-style: var(--meta-font-style);
|
||||||
font-variation-settings: var(--font-variation-settings);
|
font-variation-settings: var(--font-variation-settings);
|
||||||
margin-top: -.25rem;
|
margin-top: -.25rem;
|
||||||
}
|
}
|
||||||
wc-toot::part(main) {
|
|
||||||
color: white;
|
|
||||||
font-family: var(--meta-font-family);
|
|
||||||
font-size: var(--step--1);
|
|
||||||
}
|
|
||||||
wc-toot::part(author), wc-toot::part(author-link){
|
|
||||||
color: white;
|
|
||||||
font-style: var(--meta-font-style);
|
|
||||||
font-variation-settings: "wght" 600;
|
|
||||||
}
|
|
||||||
/* Code Fences */
|
/* Code Fences */
|
||||||
pre,
|
pre,
|
||||||
code {
|
code {
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
import './webComponents/card.js';
|
import './webComponents/card.js';
|
||||||
import './webComponents/profilePic.js';
|
import './webComponents/profilePic.js';
|
||||||
import './webComponents/comment.js';
|
import './webComponents/comment.js';
|
||||||
import './webComponents/toot.js';
|
|
||||||
|
@ -1,22 +0,0 @@
|
|||||||
const dateSuffixAdder = (date) => {
|
|
||||||
if (date > 9 && date < 20) {
|
|
||||||
return "th";
|
|
||||||
} else {
|
|
||||||
let dateString = date < 10 ? "0" + date : "" + date;
|
|
||||||
if (dateString[1] < 4 && dateString[1] > 0) {
|
|
||||||
return dateString[1] == 1 ? "st" :
|
|
||||||
dateString[1] == 2 ? "nd" :
|
|
||||||
dateString[1] == 3 ? "rd" : null;
|
|
||||||
} else {
|
|
||||||
return "th"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const monthMap = ["January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"];
|
|
||||||
|
|
||||||
const timeFormatter = (hours, minutes) => {
|
|
||||||
return `${hours < 12 ? hours : hours - 12}:${minutes < 10 ? "0" : ""}${minutes} ${hours < 12 ? "AM" : "PM"}`
|
|
||||||
}
|
|
||||||
|
|
||||||
export {dateSuffixAdder, monthMap, timeFormatter};
|
|
@ -10,6 +10,7 @@ template.innerHTML = `
|
|||||||
display: flex;
|
display: flex;
|
||||||
flex-flow: row nowrap;
|
flex-flow: row nowrap;
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
|
margin-bottom: 1em;
|
||||||
padding: var(--single-gap) 1.1rem var(--single-gap) 1.1rem;
|
padding: var(--single-gap) 1.1rem var(--single-gap) 1.1rem;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
File diff suppressed because one or more lines are too long
Loading…
x
Reference in New Issue
Block a user