feat: support nav and pages in cms

This commit is contained in:
Cory Dransfeldt 2024-07-14 19:21:04 -07:00
parent e1b0dc5243
commit f5adf0ba06
No known key found for this signature in database
35 changed files with 235 additions and 444 deletions

10
package-lock.json generated
View file

@ -1,12 +1,12 @@
{ {
"name": "coryd.dev", "name": "coryd.dev",
"version": "20.3.3", "version": "20.4.0",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "coryd.dev", "name": "coryd.dev",
"version": "20.3.3", "version": "20.4.0",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@cdransf/api-text": "^1.4.0", "@cdransf/api-text": "^1.4.0",
@ -845,9 +845,9 @@
} }
}, },
"node_modules/caniuse-lite": { "node_modules/caniuse-lite": {
"version": "1.0.30001641", "version": "1.0.30001642",
"resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001641.tgz", "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001642.tgz",
"integrity": "sha512-Phv5thgl67bHYo1TtMY/MurjkHhV4EDaCosezRXgZ8jzA/Ub+wjxAvbGvjoFENStinwi5kCyOYV3mi5tOGykwA==", "integrity": "sha512-3XQ0DoRgLijXJErLSl+bLnJ+Et4KqV1PY6JJBGAFlsNsz31zeAIncyeZfLCabHK/jtSh+671RM9YMldxjUPZtA==",
"dev": true, "dev": true,
"funding": [ "funding": [
{ {

View file

@ -1,6 +1,6 @@
{ {
"name": "coryd.dev", "name": "coryd.dev",
"version": "20.3.3", "version": "20.4.0",
"description": "The source for my personal site. Built using 11ty.", "description": "The source for my personal site. Built using 11ty.",
"type": "module", "type": "module",
"scripts": { "scripts": {

View file

@ -37,7 +37,7 @@ const fetchAlbumReleases = async () => {
timestamp: DateTime.fromISO(album['release_date']).toSeconds(), timestamp: DateTime.fromISO(album['release_date']).toSeconds(),
type: 'album-release' type: 'album-release'
} }
)).sort((a, b) => a.timestamp - b.timestamp) )).sort((a, b) => a['timestamp'] - b['timestamp'])
} }
export default async function () { export default async function () {

View file

@ -14,5 +14,5 @@ export default async function () {
}, },
}).catch() }).catch()
const pages = await res const pages = await res
return pages.results.filter((p) => p.page.includes('posts')) return pages['results'].filter((p) => p['page'].includes('posts'))
} }

View file

@ -37,7 +37,7 @@ async function fetchAllBooks() {
} }
for (const book of data) { for (const book of data) {
book.tags = await fetchTagsForBook(book.id) book.tags = await fetchTagsForBook(book['id'])
} }
books = books.concat(data) books = books.concat(data)

View file

@ -30,9 +30,9 @@ const fetchGenresWithArtists = async () => {
} }
data.forEach(genre => { data.forEach(genre => {
genre.artists = genre.artists.map(artist => ({ genre['artists'] = genre['artists'].map(artist => ({
...artist, ...artist,
country: parseCountryField(artist.country) country: parseCountryField(artist['country'])
})) }))
}) })

View file

@ -40,8 +40,8 @@ const fetchAllLinks = async () => {
if (data.length < PAGE_SIZE) fetchMore = false if (data.length < PAGE_SIZE) fetchMore = false
for (const link of data) { for (const link of data) {
link.tags = await fetchTagsForLink(link.id) link['tags'] = await fetchTagsForLink(link.id)
link.type = 'link' link['type'] = 'link'
} }
links = links.concat(data) links = links.concat(data)

View file

@ -1,30 +1,50 @@
export default async function () { import { createClient } from '@supabase/supabase-js'
return {
menu: [ const SUPABASE_URL = process.env.SUPABASE_URL
{ name: 'Posts', url: '/posts', icon: 'article'}, const SUPABASE_KEY = process.env.SUPABASE_KEY
{ name: 'Music', url: '/music', icon: 'headphones' }, const supabase = createClient(SUPABASE_URL, SUPABASE_KEY)
{ name: 'Watching', url: '/watching', icon: 'device-tv' },
{ name: 'Books', url: '/books', icon: 'books' }, const fetchAllNavigation = async () => {
{ name: 'Links', icon: 'link' }, const { data, error } = await supabase
{ name: 'About', url: '/about', icon: 'info-circle' }, .from('navigation')
{ name: 'Search', icon: 'search' }, .select(`
{ name: 'Feeds', icon: 'rss' }, *,
{ name: 'Mastodon', icon: 'brand-mastodon' }, pages(title, permalink)
], `)
footer: [
{ name: 'Uses' }, if (error) {
{ name: 'Colophon' }, console.error('Error fetching navigation data:', error)
{ name: 'Blogroll' }, return null
{ name: 'Save' },
],
social: [
{ name: 'Email', url: '/contact', icon: 'at' },
{ name: 'GitHub', url: 'https://github.com/cdransf', icon: 'brand-github' },
{ name: 'npm', url: 'https://www.npmjs.com/~cdransf', icon: 'brand-npm'},
{ name: 'Mastodon', url: 'https://social.lol/@cory', icon: 'brand-mastodon' },
{ name: 'Coffee', url: 'https://buymeacoffee.com/cory', icon: 'coffee' },
{ name: 'Now', url: '/now', icon: 'clock-hour-3' },
{ name: 'Webrings', url: '/webrings', icon: 'heart-handshake' },
],
} }
const menu = {}
data.forEach(item => {
const menuItem = item.pages ? {
title: item.pages.title,
permalink: item.pages.permalink,
icon: item.icon,
position: item.position
} : {
title: item.title,
permalink: item.permalink,
icon: item.icon,
position: item.position
}
if (!menu[item.menu_location]) {
menu[item.menu_location] = [menuItem]
} else {
menu[item.menu_location].push(menuItem)
}
})
Object.keys(menu).forEach(location => {
menu[location].sort((a, b) => a.position - b.position)
})
return menu
} }
export default async function () {
return await fetchAllNavigation()
}

82
src/_data/pages.js Normal file
View file

@ -0,0 +1,82 @@
import { createClient } from '@supabase/supabase-js'
const SUPABASE_URL = process.env.SUPABASE_URL
const SUPABASE_KEY = process.env.SUPABASE_KEY
const supabase = createClient(SUPABASE_URL, SUPABASE_KEY)
const PAGE_SIZE = 50
const fetchBlockData = async (collection, itemId) => {
const { data, error } = await supabase
.from(collection)
.select(collection === 'hero' ? '*, image(filename_disk)' : '*')
.eq('id', itemId)
.single()
if (error) {
console.error(`Error fetching data from ${collection} for item ${itemId}:`, error)
return null
}
return data
}
const fetchBlocksForPage = async (pageId) => {
const { data, error } = await supabase
.from('pages_blocks')
.select('collection, item')
.eq('pages_id', pageId)
if (error) {
console.error(`Error fetching blocks for page ${pageId}:`, error)
return []
}
const blocks = await Promise.all(data.map(async block => {
const blockData = await fetchBlockData(block.collection, block.item)
return {
type: block['collection'],
...blockData
}
}))
return blocks
}
const fetchAllPages = async () => {
let pages = []
let page = 0
let fetchMore = true
while (fetchMore) {
const { data, error } = await supabase
.from('pages')
.select(`
*,
open_graph_image(filename_disk)
`)
.range(page * PAGE_SIZE, (page + 1) * PAGE_SIZE - 1)
if (error) {
console.error('Error fetching pages:', error)
return pages
}
if (data.length < PAGE_SIZE) fetchMore = false
for (const page of data) {
page['blocks'] = await fetchBlocksForPage(page['id'])
if (page['open_graph_image']) page['open_graph_image'] = page['open_graph_image']['filename_disk']
pages.push(page)
}
page++
}
return pages
}
export default async function () {
return await fetchAllPages()
}

View file

@ -26,7 +26,7 @@ const fetchAllRobots = async () => {
if (data.length < PAGE_SIZE) break if (data.length < PAGE_SIZE) break
} }
return robots.map(robot => robot.user_agent) return robots.map(robot => robot['user_agent'])
} }
export default async function () { export default async function () {

View file

@ -8,5 +8,5 @@ export default async function () {
}).catch() }).catch()
const status = await res const status = await res
return status.response['statuses'][0] return status['response']['statuses'][0]
} }

View file

@ -47,20 +47,20 @@ const fetchAllShows = async () => {
const prepareShowData = (show) => { const prepareShowData = (show) => {
return { return {
...show, ...show,
image: show.art?.filename_disk ? `/${show.art.filename_disk}` : '', image: show['art']?.['filename_disk'] ? `/${show['art']['filename_disk']}` : '',
backdrop: show.backdrop?.filename_disk ? `/${show.backdrop.filename_disk}` : '' backdrop: show['backdrop']?.['filename_disk'] ? `/${show['backdrop']['filename_disk']}` : ''
} }
} }
const prepareEpisodeData = (show) => { const prepareEpisodeData = (show) => {
return show.episodes.map(episode => ({ return show['episodes'].map(episode => ({
...episode, ...episode,
show_title: show.title, show_title: show['title'],
show_tmdb_id: show.tmdb_id, show_tmdb_id: show['tmdb_id'],
collected: show.collected, collected: show['collected'],
favorite: show.favorite, favorite: show['favorite'],
image: show.image, image: show['image'],
backdrop: show.backdrop backdrop: show['backdrop']
})) }))
} }
@ -76,15 +76,15 @@ export default async function () {
const showEpisodesMap = {} const showEpisodesMap = {}
episodes.forEach(episode => { episodes.forEach(episode => {
const showTitle = episode.show_title const showTitle = episode['show_title']
const showTmdbId = episode.show_tmdb_id const showTmdbId = episode['show_tmdb_id']
const episodeNumber = episode.episode_number const episodeNumber = episode['episode_number']
const seasonNumber = episode.season_number const seasonNumber = episode['season_number']
const lastWatchedAt = episode.last_watched_at const lastWatchedAt = episode['last_watched_at']
const collected = episode.collected const collected = episode['collected']
const favorite = episode.favorite const favorite = episode['favorite']
const image = episode.image const image = episode['image']
const backdrop = episode.backdrop const backdrop = episode['backdrop']
if (!showEpisodesMap[showTmdbId]) { if (!showEpisodesMap[showTmdbId]) {
showEpisodesMap[showTmdbId] = { showEpisodesMap[showTmdbId] = {
@ -115,16 +115,16 @@ export default async function () {
}) })
}) })
const sortedShows = Object.values(showEpisodesMap).sort((a, b) => new Date(b.episodes[0].lastWatchedAt) - new Date(a.episodes[0].lastWatchedAt)) const sortedShows = Object.values(showEpisodesMap).sort((a, b) => new Date(b.episodes[0]['lastWatchedAt']) - new Date(a.episodes[0]['lastWatchedAt']))
const episodeData = [] const episodeData = []
sortedShows.forEach(show => { sortedShows.forEach(show => {
const startingEpisode = show.episodes[show.episodes.length - 1].episode const startingEpisode = show['episodes'][show['episodes'].length - 1]['episode']
const startingSeason = show.episodes[show.episodes.length - 1].season const startingSeason = show['episodes'][show['episodes'].length - 1]['season']
const endingEpisode = show.episodes[0].episode const endingEpisode = show['episodes'][0].episode
const endingSeason = show.episodes[0].season const endingSeason = show['episodes'][0].season
if (show.episodes.length > 1) { if (show['episodes'].length > 1) {
episodeData.push({ episodeData.push({
name: show.title, name: show.title,
url: `/watching/shows/${show.tmdbId}`, url: `/watching/shows/${show.tmdbId}`,
@ -141,11 +141,11 @@ export default async function () {
backdrop: show.backdrop backdrop: show.backdrop
}) })
} else { } else {
const singleEpisode = show.episodes[0] const singleEpisode = show['episodes'][0]
singleEpisode.collected = show.collected singleEpisode.collected = show['collected']
singleEpisode.favorite = show.favorite singleEpisode.favorite = show['favorite']
singleEpisode.image = show.image singleEpisode.image = show['image']
singleEpisode.backdrop = show.backdrop singleEpisode.backdrop = show['backdrop']
episodeData.push(singleEpisode) episodeData.push(singleEpisode)
} }
}) })
@ -153,12 +153,12 @@ export default async function () {
return episodeData return episodeData
} }
const favoriteShows = shows.filter(show => show.favorite) const favoriteShows = shows.filter(show => show['favorite'])
return { return {
shows, shows,
watchHistory: formatEpisodeData(episodes), watchHistory: formatEpisodeData(episodes),
recentlyWatched: formatEpisodeData(episodes.slice(0, 225)), recentlyWatched: formatEpisodeData(episodes.slice(0, 225)),
favorites: formatEpisodeData(favoriteShows.flatMap(prepareEpisodeData)).sort((a, b) => a.name.localeCompare(b.name)) favorites: formatEpisodeData(favoriteShows.flatMap(prepareEpisodeData)).sort((a, b) => a['name'].localeCompare(b['name']))
} }
} }

View file

@ -3,5 +3,5 @@ layout: base
--- ---
<div class="main-wrapper"> <div class="main-wrapper">
<main>{{ content }}</main> <main>{{ content }}</main>
{% render "partials/footer.liquid", page:page, nav:nav, updated:updated %} {% render "partials/footer.liquid", page:page, nav:nav, updated:page.updated %}
</div> </div>

View file

@ -0,0 +1,19 @@
<img
srcset="
https://cdn.coryd.dev/{{ image }}?class=bannersm 256w,
https://cdn.coryd.dev/{{ image }}?class=bannermd 512w,
https://cdn.coryd.dev/{{ image }}?class=bannerbase 1024w,
https://cdn.coryd.dev/{{ image }}?class=bannerlg 2048w
"
sizes="(max-width: 450px) 256px,
(max-width: 850px) 512px,
(max-width: 1000px) 1024px,
2048px"
src="https://cdn.coryd.dev/{{ image }}?class=bannerlg"
alt="{{ alt }}"
class="image-banner"
loading="eager"
decoding="async"
width="1080"
height="720"
/>

View file

@ -7,5 +7,11 @@
{% render "partials/banners/npm.liquid", url:block.url, command:block.command %} {% render "partials/banners/npm.liquid", url:block.url, command:block.command %}
{% elsif block.type == 'rss_banner' %} {% elsif block.type == 'rss_banner' %}
{% render "partials/banners/rss.liquid", url:block.url, text:block.text %} {% render "partials/banners/rss.liquid", url:block.url, text:block.text %}
{% elsif block.type == 'hero' %}
{% render "partials/blocks/hero.liquid", image:block.image.filename_disk, alt:block.alt_text %}
{% elsif block.type == 'markdown' %}
{{ block.text | markdown }}
{% elsif block.type == 'addon_links' %}
<hr />{% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %}
{% endif %} {% endif %}
{% endfor %} {% endfor %}

View file

@ -1,14 +1,16 @@
<footer{% unless updated %} style="margin-top:var(--sizing-3xl)"{% endunless %}> <footer{% unless updated %} style="margin-top:var(--sizing-3xl)"{% endunless %}>
{% if updated %}<p class="explainer text-small text-centered"><em>This page was last updated on {{ updated | date: "%B %-d, %-I:%M%p", "America/Los_Angeles" }}.</em></p>{% endif %} {% if updated %}
<p class="explainer text-small text-centered"><em>This page was last updated on {{ updated | date: "%B %-d, %-I:%M%p", "America/Los_Angeles" }}.</em></p>
{% endif %}
<nav aria-label="Social icons" class="social flex-centered justify-centered text-centered"> <nav aria-label="Social icons" class="social flex-centered justify-centered text-centered">
{% for link in nav.social %} {% for link in nav.footer_icons %}
{% render "partials/nav/linked-icon.liquid", name:link.name, link:link.url, icon:link.icon %} {% render "partials/nav/linked-icon.liquid", title:link.title, url:link.permalink, icon:link.icon %}
{% endfor %} {% endfor %}
</nav> </nav>
<nav aria-label="Secondary site navigation" class="sub-pages flex-centered justify-centered text-centered"> <nav aria-label="Secondary site navigation" class="sub-pages flex-centered justify-centered text-centered">
{% for link in nav.footer %} {% for link in nav.footer_text %}
{% render "partials/nav/link.liquid", page:page, link:link.name, icon:link.icon %} {% render "partials/nav/link.liquid", page:page, title:link.title, url:link.permalink, icon:link.icon %}
{% if not forloop.last %}<span>/</span>{% endif %} {% if not forloop.last %}<span>/</span>{% endif %}
{% endfor %} {% endfor %}
</nav> </nav>
</footer> </footer>

View file

@ -1,20 +1,20 @@
{%- assign categoryUrl = "/" | append: link | downcase -%} {%- assign categoryUrl = url | downcase -%}
{% if categoryUrl | isLinkActive: page.url %} {% if categoryUrl | isLinkActive: page.url %}
<span class="active {{ class }}" aria-current="page"> <span class="active {{ class }}" aria-current="page">
{% if icon %} {% if icon %}
{% tablericon icon link %} {% tablericon icon title %}
<span>{{ link }}</span> <span>{{ title }}</span>
{% else %} {% else %}
{{ link }} {{ title }}
{% endif %} {% endif %}
</span> </span>
{% else %} {% else %}
<a class="{% if icon %}{{ icon | downcase }} icon {% endif %}{{ class }}" href="{{ categoryUrl }}" tabindex="0"> <a class="{% if icon %}{{ icon | downcase }} icon {% endif %}{{ class }}" href="{{ categoryUrl }}" tabindex="0">
{% if icon %} {% if icon %}
{% tablericon icon link %} {% tablericon icon title %}
<span>{{ link }}</span> <span>{{ title }}</span>
{% else %} {% else %}
{{ link }} {{ title }}
{% endif %} {% endif %}
</a> </a>
{% endif %} {% endif %}

View file

@ -1,8 +1,8 @@
<a <a
class="{{ icon }}" class="{{ icon }}"
href="{{ link | downcase }}" href="{{ url | downcase }}"
rel="me" rel="me"
title="{{ name }}" title="{{ title }}"
tabindex="0"> tabindex="0">
{% tablericon icon name %} {% tablericon icon title %}
</a> </a>

View file

@ -9,8 +9,8 @@
<div class="menu-open" aria-hidden="true">{% tablericon "x" "Menu open" %}</div> <div class="menu-open" aria-hidden="true">{% tablericon "x" "Menu open" %}</div>
</label> </label>
<ul class="menu-primary" aria-label="Primary site navigation" id="primary-navigation"> <ul class="menu-primary" aria-label="Primary site navigation" id="primary-navigation">
{% for link in nav.menu %} {% for link in nav.primary %}
<li>{% render "partials/nav/link.liquid", page:page, link:link.name, icon:link.icon %}</li> <li>{% render "partials/nav/link.liquid", page:page, title:link.title, url:link.permalink, icon:link.icon %}</li>
{% endfor %} {% endfor %}
</ul> </ul>
{% render "partials/nav/theme-toggle.liquid" %} {% render "partials/nav/theme-toggle.liquid" %}

View file

@ -1,13 +0,0 @@
---
title: Error
description: Nah, that's no good.
layout: default
permalink: /1000.html
---
<div class="text-centered">
<h2 class="page-header">{{ title }}</h2>
<p>Nah, that's no good.</p>
<hr />
::CLOUDFLARE_ERROR_1000S_BOX::
</div>
<script>document.addEventListener('DOMContentLoaded', function () { plausible('1000', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,34 +0,0 @@
---
title: 404
description: What kind of idiots do you have working here? Hurry up and skip out on the room service bill!
layout: default
permalink: /404.html
image: https://cdn.coryd.dev/{{ globals.page_404 }}
---
<img
srcset="
https://cdn.coryd.dev/{{ globals.page_404 }}?class=bannersm 256w,
https://cdn.coryd.dev/{{ globals.page_404 }}?class=bannermd 512w,
https://cdn.coryd.dev/{{ globals.page_404 }}?class=bannerbase 1024w,
https://cdn.coryd.dev/{{ globals.page_404 }}?class=bannerlg 2048w
"
sizes="(max-width: 450px) 256px,
(max-width: 850px) 512px,
(max-width: 1000px) 1024px,
2048px"
src="https://cdn.coryd.dev/{{ globals.page_404 }}?class=bannerlg"
alt="{{ alt }}"
class="image-banner"
loading="eager"
decoding="async"
width="1080"
height="720"
/>
<div class="text-centered">
<h2>{{ title }}</h2>
<p>What kind of idiots do you have working here?</p>
<p><a href="/">Hurry up and skip out on the room service bill!</a></p>
</div>
<hr />
{% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %}
<script>document.addEventListener('DOMContentLoaded', function () { plausible('404', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,13 +0,0 @@
---
title: Error
description: Nah, that's no good.
layout: default
permalink: /500.html
---
<div class="text-centered">
<h2 class="page-header">{{ title }}</h2>
<p>Nah, that's no good.</p>
<hr />
::CLOUDFLARE_ERROR_500S_BOX::
</div>
<script>document.addEventListener('DOMContentLoaded', function () { plausible('500', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,34 +0,0 @@
---
title: Broken
description: Hey! That's broken. Try again later.
layout: default
permalink: /broken.html
image: https://cdn.coryd.dev/{{ globals.page_broken }}
---
<img
srcset="
https://cdn.coryd.dev/{{ globals.page_broken }}?class=bannersm 256w,
https://cdn.coryd.dev/{{ globals.page_broken }}?class=bannermd 512w,
https://cdn.coryd.dev/{{ globals.page_broken }}?class=bannerbase 1024w,
https://cdn.coryd.dev/{{ globals.page_broken }}?class=bannerlg 2048w
"
sizes="(max-width: 450px) 256px,
(max-width: 850px) 512px,
(max-width: 1000px) 1024px,
2048px"
src="https://cdn.coryd.dev/{{ globals.page_broken }}?class=bannerlg"
alt="{{ alt }}"
class="image-banner"
loading="eager"
decoding="async"
width="1080"
height="720"
/>
<div class="text-centered">
<h2>{{ title }}</h2>
<p>Hey! That's broken. Try again later.</p>
<p><a href="/">Go read something instead.</a></p>
</div>
<hr />
{% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %}
<script>document.addEventListener('DOMContentLoaded', function () { plausible('Broken', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,13 +0,0 @@
---
title: Error
description: Nah, that's no good.
layout: default
permalink: /captcha.html
---
<div class="text-centered">
<h2 class="page-header">{{ title }}</h2>
<p>Nah, that's no good.</p>
<hr />
::CAPTCHA_BOX::
</div>
<script>document.addEventListener('DOMContentLoaded', function () { plausible('Captcha', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,11 +0,0 @@
---
title: Error
description: Nah, that's no good.
layout: default
permalink: /error.html
---
<div class="text-centered">
<h2 class="page-header">{{ title }}</h2>
<p>Nah, that's no good.</p>
</div>
<script>document.addEventListener('DOMContentLoaded', function () { plausible('error', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,13 +0,0 @@
---
title: Error
description: Nah, that's no good.
layout: default
permalink: /js-challenge.html
---
<div class="text-centered">
<h2 class="page-header">{{ title }}</h2>
<p>Nah, that's no good.</p>
<hr />
::IM_UNDER_ATTACK_BOX::
</div>
<script>document.addEventListener('DOMContentLoaded', function () { plausible('JS challenge', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,34 +0,0 @@
---
title: Not allowed
description: Sometimes mildly inconvenient things happen to people. That's not allowed.
layout: default
permalink: /not-allowed.html
image: https://cdn.coryd.dev/{{ globals.page_not_allowed }}
---
<img
srcset="
https://cdn.coryd.dev/{{ globals.page_not_allowed }}?class=bannersm 256w,
https://cdn.coryd.dev/{{ globals.page_not_allowed }}?class=bannermd 512w,
https://cdn.coryd.dev/{{ globals.page_not_allowed }}?class=bannerbase 1024w,
https://cdn.coryd.dev/{{ globals.page_not_allowed }}?class=bannerlg 2048w
"
sizes="(max-width: 450px) 256px,
(max-width: 850px) 512px,
(max-width: 1000px) 1024px,
2048px"
src="https://cdn.coryd.dev/{{ globals.page_not_allowed }}?class=bannerlg"
alt="{{ alt }}"
class="image-banner"
loading="eager"
decoding="async"
width="1080"
height="720"
/>
<div class="text-centered">
<h2>{{ title }}</h2>
<p>Sometimes mildly inconvenient things happen to people.</p>
<p><a href="/">Go read something instead.</a></p>
</div>
<hr />
{% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %}
<script>document.addEventListener('DOMContentLoaded', function () { plausible('Not allowed', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,34 +0,0 @@
---
title: Rate limit
description: Hey! Stop that. Try again later.
layout: default
permalink: /rate-limit.html
image: https://cdn.coryd.dev/{{ globals.page_stop }}
---
<img
srcset="
https://cdn.coryd.dev/{{ globals.page_stop }}?class=bannersm 256w,
https://cdn.coryd.dev/{{ globals.page_stop }}?class=bannermd 512w,
https://cdn.coryd.dev/{{ globals.page_stop }}?class=bannerbase 1024w,
https://cdn.coryd.dev/{{ globals.page_stop }}?class=bannerlg 2048w
"
sizes="(max-width: 450px) 256px,
(max-width: 850px) 512px,
(max-width: 1000px) 1024px,
2048px"
src="https://cdn.coryd.dev/{{ globals.page_stop }}?class=bannerlg"
alt="{{ alt }}"
class="image-banner"
loading="eager"
decoding="async"
width="1080"
height="720"
/>
<div class="text-centered">
<h2>{{ title }}</h2>
<p>Hey! Stop that. Try again later.</p>
<p><a href="/">Go read something instead.</a></p>
</div>
<hr />
{% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %}
<script>document.addEventListener('DOMContentLoaded', function () { plausible('Rate limit', { props: { path: document.location.pathname } }); });</script>

View file

@ -1,15 +0,0 @@
---
title: Feeds
layout: page
permalink: /feeds.html
description: Content feeds exposed by and generated from my site.
---
<h2 class="page-header">{{ title }}</h2>
These are web feeds, also known as RSS feeds. Subscribe by copying a URL into your newsreader.
- Posts ([RSS](https://feedpress.me/coryd) or [JSON](https://feedpress.me/coryd.json)): posts from my site.
- Links ([RSS](https://feedpress.me/coryd-links) or [JSON](https://feedpress.me/coryd-links.json)): links I've liked.
- Books ([RSS](https://feedpress.me/coryd-books) or [JSON](https://feedpress.me/coryd-books.json)): books I've finished recently.
- Movies ([RSS](https://feedpress.me/coryd-movies) or [JSON](https://feedpress.me/coryd-books.json)): movies I've watched recently.
- All ([RSS](https://feedpress.me/coryd-all) or [JSON](https://feedpress.me/coryd-all.json)): all of the posts and activity from my site.

View file

@ -1,21 +0,0 @@
---
title: Now
layout: default
permalink: /now.html
description: See what I'm doing now.
updated: 2024-05-18T15:49:00Z
---
<h2 class="page-header">{{ title }}</h2>
<h3>Family</h3>
<p>Hanging out with and spending as much time as possible with my <strong class="highlight-text">lovely wife, kids and rescue dogs (we've got a guinea pig too).</strong></p>
<h3><a href="https://coryd.dev/posts/2024/on-getting-tattooed/">Getting tattooed</a></h3>
<p>It's been an ongoing process of recovering from sessions, resuming my exercising routine and returning for the next session. I have two chest pieces, a new piece above my left knee and we've been working away on my right arm.</p>
<h3><a href="https://github.com/cdransf">Making</a></h3>
<p><strong class="highlight-text">Scrobbling and data ownership:</strong> <a href="https://coryd.dev/posts/2024/building-a-scrobbler-using-plex-webhooks-edge-functions-and-blob-storage/">I've been building</a> and <a href="https://coryd.dev/posts/2024/improving-my-self-hosted-scrobbling-implementation/">improving</a> my self-hosted scrobbling solution. <a href="https://coryd.dev/posts/2024/adventures-in-self-hosting-data/">I've been working to bring my movie, TV and reading data in-house too.</a></p>
<p><strong class="highlight-text">Ad and tracker-blocking:</strong> I've been updating and maintaining a list of ad and tracker-blocking tools <a href="https://github.com/cdransf/awesome-adblock">that I'm happy to take contributions to.</a> Modern advertising is awful and intrusive. <a href="https://coryd.dev/posts/2023/i-block-ads/">Do what you can to block it.</a></p>
<p><strong class="highlight-text">Fiddling with simple web components:</strong> I've been working on and incorporating some web components into this site. I've built a <a href="https://github.com/cdransf/theme-toggle">theme toggle</a>, <a href="https://github.com/cdransf/select-pagination">select-based pagination</a> and <a href="https://github.com/cdransf/api-text">one that loads text from an API</a> (used on my home page).</p>
<p><strong class="highlight-text">Encouraging folks to block AI web crawlers:</strong> <a href="https://coryd.dev/posts/2024/go-ahead-and-block-ai-web-crawlers/">they're insidious, abusive and should be blocked.</a> <a href="https://github.com/ai-robots-txt/ai.robots.txt">I'm happy to take contributions, once again,</a> as we encourage folks to block them as best they can.</p>
<h3>Mentoring</h3>
<p>I've been mentoring through <a href="https://www.underdogdevs.org">Underdog Devs</a> on and off for a few years now and find it incredibly rewarding.</p>
<hr />
<p>This is a <a href="https://nownownow.com/about">now page</a>, and if you have your own site, <a href="https://nownownow.com/about">you should make one too</a>.</p>

12
src/pages/main/page.html Normal file
View file

@ -0,0 +1,12 @@
---
layout: default
pagination:
data: pages
size: 1
alias: page
description: "{{ page.description | default: globals.site_description }}"
permalink: "{{ page.permalink }}/index.html"
image: "{{ page.open_graph_image | prepend: 'https://cdn.coryd.dev/' | default: globals.meta_data.opengraph_default }}"
updated: {{ page.updated | default: null }}
---
{% render "partials/blocks/index.liquid", blocks:page.blocks, collections:collections, links:links %}

View file

@ -39,7 +39,7 @@ schema: blog
/> />
{%- endif -%} {%- endif -%}
{{ post.content | markdown }} {{ post.content | markdown }}
{% render "partials/posts/blocks.liquid", blocks:post.blocks %} {% render "partials/blocks/index.liquid", blocks:post.blocks %}
</div> </div>
</article> </article>
{% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %} {% render "partials/widgets/addon-links.liquid", popularPosts:collections.popularPosts, links:links %}

View file

@ -1,19 +0,0 @@
---
title: Colophon
layout: page
permalink: /colophon.html
description: The tools I use to build and maintain this site.
---
<h2 class="page-header">{{ title }}</h2>
This site is built and maintained using a number of tools.
- The frontend of the site is built using [11ty](https://www.11ty.dev) and the source for that is openly available on [GitHub](https://github.com/cdransf/coryd.dev).
- The frontend is hosted on [Cloudflare Pages](https://pages.cloudflare.com) and rebuilds hourly.
- Posts, links and media data (music, watching and books) are stored at [Supabase](https://supabase.com) and managed via a [Directus](https://directus.io) instance hosted at <a class="plausible-event-name=DigitalOcean+referral" href="https://m.do.co/c/3635bf99aee2">DigitalOcean</a>.
- My contact form is submitted to [Supabase](https://supabase.com) using a Cloudflare worker and entries are available to read in [Directus](https://directus.io).
- Posts are composed in [Obsidian](https://obsidian.md) before being saved in [Directus](https://directus.io).
- Images are hosted on [Backblaze](https://backblaze.com) B2 and served by <a class="plausible-event-name=bunny.net+referral" href="https://bunny.net?ref=revw3mehej">bunny.net</a>.
- I use <a class="plausible-event-name=DNSimple+referral" href="https://dnsimple.com/r/3a7cbb9e15df8f">DNSimple</a> to register my domains and [DNSControl](https://dnscontrol.org) to configure and manage records.
- I use [Plausible](https://plausible.io) for analytics.
- <a class="plausible-event-name=Feedpress+referral" href="https://feedpress.com/?affid=34370">Feedpress</a> helps normalize my feeds and provides lightweight feed insights.

View file

@ -1,20 +0,0 @@
---
title: Save
layout: page
permalink: /save.html
description: Save a little bit on services that I also use.
---
<h2 class="page-header">{{ title }}</h2>
Referral links for services I use. I save some money, and you do as well if you choose to use them.
<ul class="link-list">
<li><a class="plausible-event-name=Proton+referral" href="https://pr.tn/ref/X775YX40Z50G">Proton</a></li>
<li><a class="plausible-event-name=NextDNS+referral" href="https://nextdns.io/?from=m56mt3z6">NextDNS</a></li>
<li><a class="plausible-event-name=DNSimple+referral" href="https://dnsimple.com/r/3a7cbb9e15df8f">DNSimple</a></li>
<li><a class="plausible-event-name=bunny.net+referral" href="https://bunny.net?ref=revw3mehej">bunny.net</a></li>
<li><a class="plausible-event-name=Feedpress+referral" href="https://feedpress.com/?affid=34370">Feedpress</a></li>
<li><a class="plausible-event-name=Matter+referral" href="https://web.getmatter.com/referral/ss6td795">Matter</a></li>
<li><a class="plausible-event-name=DigitalOcean+referral" href="https://m.do.co/c/3635bf99aee2">DigitalOcean</a></li>
<li><a class="plausible-event-name=Garbage+Day+referral" href="https://www.garbageday.email/subscribe?ref=4JeD4bFKQE">Garbage Day newsletter</a></li>
</ul>

View file

@ -1,76 +0,0 @@
---
title: Uses
layout: page
permalink: /uses.html
description: Software, tools and services that I use regularly.
---
<h2 class="page-header">{{ title }}</h2>
Software and services that I use for work and my own enjoyment.
<h3>Computer setup</h3>
- Midnight MacBook Air (2022 - M2)
- 27" Dell Monitor
- Apple Magic Keyboard with Touch ID + number pad
- Apple Magic Trackpad
- Homepod Mini for audio
<h3>Desk and chair</h3>
- My desk is a custom made corner desk I bought on Etsy years ago. It's sturdy, has tons of storage and finally arrived with a chip on the surface to add some character after the first one got lost by the freight company on the journey from Michigan to Los Angeles. I probably _wouldn't_ repeat the experience, but I do love the desk.
- [Herman Miller Aeron chair](https://www.hermanmiller.com/products/seating/office-chairs/aeron-chairs/): it's pretty comfortable and makes swiveling around my corner desk easy.
<h3>macOS + iOS</h3>
- [Vivaldi](https://vivaldi.net): a flexible, reliable and privacy focused browser.
- [Fantastical](https://flexibits.com/fantastical): a powerful calendar app that continues to outpace Apple's calendar app.
- [Obsidian](https://obsidian.md): fast and configurable (or minimal) as you'd like, I use it for all of my notes and writing.
- [Plexamp](https://www.plex.tv/plexamp/): Plex's flexible and delightful music player.
- [Ivory](https://tapbots.com/ivory/): the best, most polished Mastodon client for macOS and iOS.
- [Reeder](https://reeder.app): flexible and universal — it makes it easy to triage my feeds and save things over to Matter.
- [Parcel](https://parcelapp.net): the most flexible and reliable package tracker for Apple's ecosystem.
- [Flighty](https://flightyapp.com): I don't travel a ton but Flighty makes doing so a fair bit less stressful.
<h3>iOS</h3>
- [Matter](https://getmatter.com): my favorite read it later service. It's thoughtful, beautifully designed and it's text to speech features are the best out there.
- [status.log](https://apps.apple.com/us/app/status-log/id6444921793): a fantastic client for status.lol that provides an outstanding native experience for the service.
- [Working Copy](https://workingcopy.app): an incredible, fully-featured git client.
- [Secure Shellfish](https://secureshellfish.app): the _best_ ssh client for iOS, complete with Codeapaces support and Files.app integration.
- [Runestone](https://runestone.app): a superb, lightweight text editor for iOS that works beautifully with Working Copy and Secure Shellfish via Files.app.
- [FontCase](https://apps.apple.com/us/app/fontcase-manage-your-type/id1205074470): for managing/installing fonts.
<h3>macOS</h3>
- [Rectangle](https://rectangleapp.com): to quickly move around/organize/snap application windows. Using a Mac without it now feels like it's broken.
- [AirBuddy](https://v2.airbuddy.app): finer-grained control over AirPods and other wireless devices.
- [Meta](https://www.nightbirdsevolve.com/meta): the _best_ utility for tagging and organizing music files on macOS.
- [Permute](https://software.charliemonroe.net/permute): a useful utility for quickly converting files to different formats.
- [noTunes](https://github.com/tombonez/noTunes): a lightweight utility that prevents Music.app from launching. It also allows you to set a new default music player.
<h3>Dev tools</h3>
- [VS Code](https://code.visualstudio.com): it's the industry standard — for better or worse.
- [Catppuccin](https://github.com/catppuccin): nearly as ubiquitous as Dracula but lighter and more playful, I've started using this theme wherever I'm looking at/reading/writing code.
- [Mono Lisa](https://monolisa.dev): a relatively new find, I've been enjoying how pleasant and readable this font is (and have even gone so far as to install it on iOS via [FontCase](https://apps.apple.com/us/app/fontcase-manage-your-type/id1205074470)).
<h3>Services</h3>
- <a class="plausible-event-name=Proton+referral" href="https://pr.tn/ref/X775YX40Z50G">Proton</a>: the premier encrypted mail service with a number of other features like calendars and a reliable VPN.
- <a class="plausible-event-name=NextDNS+referral" href="https://nextdns.io/?from=m56mt3z6">NextDNS</a>: a privacy-focused, set it and forget it DNS service. I use their security features on my home network and a profile with strict ad-blocking rules on all of my devices.
- <a class="plausible-event-name=DNSimple+referral" href="https://dnsimple.com/r/3a7cbb9e15df8f">DNSimple</a>: a robust, user-friendly DNS provider and registrar. I moved my domains here after my old provider was acquired.
- [Cloudflare](https://cloudfllare.com): I use their pages hosting, workers and myriad other features.
- [Supabase](https://supabase.com): an easy to use and robust database platform built on postgres.
- [Directus](https://directus.io): an extremely flexible backend application that I use as the content management system for this site.
- <a class="plausible-event-name=bunny.net+referral" href="https://bunny.net?ref=revw3mehej">bunny.net</a>: an affordable and altogether user-friendly CDN. I host the images for my site with them and use their optimizer/transforms heavily.
- [Plausible](https://plausible.io): lightweight, privacy-friendly analytics.
- <a class="plausible-event-name=Feedpress+referral" href="https://feedpress.com/?affid=34370">Feedpress</a>: they've been around for a while now and don't change much (nor do they need to), but look no further for reliable, helpful feed analytics.
- [Feedbin](https://feedbin.com): performant, open and super reliable RSS.
- <a class="plausible-event-name=Matter+referral" href="https://web.getmatter.com/referral/ss6td795">Matter</a>: a newer read it later service, but a rapidly developed and extremely powerful one. It's text to speech features are _excellent_.
- [forwardemail.net](https://forwardemail.net): a simple and reliable service for forwarding and routing emails from a few of the domains I own.
- [Arq](https://arqbackup.com): It backs up my MacBook Air and attached storage drive to Backblaze B2 and a Hetzner storage box.
<hr />
Check out [uses.tech](https://uses.tech) for more lists like this one.