feat: add tina CMS; layouts

This commit is contained in:
Aaron Yarborough 2025-03-28 21:24:09 +00:00
parent 662a461c3c
commit 6fc63fd50c
6 changed files with 79 additions and 72 deletions

View file

@ -14,6 +14,11 @@
"test": "jest --verbose --passWithNoTests", "test": "jest --verbose --passWithNoTests",
"lint": "next lint" "lint": "next lint"
}, },
"standard": {
"ignore": [
"/tina/**/*"
]
},
"dependencies": { "dependencies": {
"@highlightjs/cdn-assets": "^11.11.1", "@highlightjs/cdn-assets": "^11.11.1",
"date-fns": "^4.1.0", "date-fns": "^4.1.0",

View file

@ -3,7 +3,7 @@ import { NextSeo } from 'next-seo'
import Link from 'next/link' import Link from 'next/link'
import React from 'react' import React from 'react'
function Book({ attributes, html }) { function Book ({ attributes, html }) {
return ( return (
<> <>
<h1>{attributes.title}<br /><small>by {attributes.author}</small></h1> <h1>{attributes.title}<br /><small>by {attributes.author}</small></h1>
@ -23,7 +23,7 @@ function Book({ attributes, html }) {
<div> <div>
<Link href='./'>Back...</Link> <Link href='./'>Back...</Link>
{attributes.pubdate && <p>{formatDate(attributes.pubdate)}</p>} {attributes.pubdate && <p>{formatDate(attributes.pubdate)}</p>}
<div data-test='content' dangerouslySetInnerHTML={{ __html: html || "<p>(no review)</p>" }} /> <div data-test='content' dangerouslySetInnerHTML={{ __html: html || '<p>(no review)</p>' }} />
</div> </div>
</article> </article>
</> </>

View file

@ -1,13 +1,15 @@
import style from './BookListItem.module.css' import style from './BookListItem.module.css'
import Link from 'next/link' import Link from 'next/link'
export default function BookListItem({ href, title, author, stars, readDate, url, thumbnailUrl, tags, review }) { export default function BookListItem ({ href, title, author, stars, readDate, url, thumbnailUrl, tags, review }) {
return ( return (
<div className={style.item}> <div className={style.item}>
<Link href={href}> <Link href={href}>
<div className={style['thumb']} style={{ <div
backgroundImage: `url(${thumbnailUrl ?? '/img/book-placeholder.jpg'})` className={style.thumb} style={{
}}></div> backgroundImage: `url(${thumbnailUrl ?? '/img/book-placeholder.jpg'})`
}}
/>
</Link> </Link>
<div> <div>
<h2 className={style.heading}> <h2 className={style.heading}>

View file

@ -56,7 +56,7 @@ export function getStaticEntries (contentPath) {
const directoryItems = fs.readdirSync(contentPath, { withFileTypes: true }) const directoryItems = fs.readdirSync(contentPath, { withFileTypes: true })
return directoryItems.map((dirent) => return directoryItems.map((dirent) =>
getMarkdownEntry(`${dirent.path}/${dirent.name}`) getMarkdownEntry(`${dirent.path}/${dirent.name}`)
); )
} }
export function getContentTags (contentPath) { export function getContentTags (contentPath) {

View file

@ -10,18 +10,18 @@ export const getStaticProps = () => {
const bookEntries = getStaticEntries('./content/books') const bookEntries = getStaticEntries('./content/books')
.sort((a, b) => { .sort((a, b) => {
return b.attributes.readDate - a.attributes.readDate return b.attributes.readDate - a.attributes.readDate
}); })
return { return {
props: { props: {
// Silliness to make sure dates don't get passed to the // Silliness to make sure dates don't get passed to the
// page function below as [object Object] // page function below as [object Object]
bookEntries: JSON.parse(JSON.stringify(bookEntries)) bookEntries: JSON.parse(JSON.stringify(bookEntries))
} }
} }
} }
export default function Library({ bookEntries }) { export default function Library ({ bookEntries }) {
return ( return (
<DefaultLayout> <DefaultLayout>
<NextSeo <NextSeo
@ -44,4 +44,4 @@ export default function Library({ bookEntries }) {
</section> </section>
</DefaultLayout> </DefaultLayout>
) )
} }

View file

@ -1,11 +1,11 @@
import { defineConfig } from "tinacms"; import { defineConfig } from 'tinacms'
// Your hosting provider likely exposes this as an environment variable // Your hosting provider likely exposes this as an environment variable
const branch = const branch =
process.env.GITHUB_BRANCH || process.env.GITHUB_BRANCH ||
process.env.VERCEL_GIT_COMMIT_REF || process.env.VERCEL_GIT_COMMIT_REF ||
process.env.HEAD || process.env.HEAD ||
"main"; 'main'
export default defineConfig({ export default defineConfig({
branch, branch,
@ -16,121 +16,121 @@ export default defineConfig({
token: process.env.TINA_TOKEN, token: process.env.TINA_TOKEN,
build: { build: {
outputFolder: "admin", outputFolder: 'admin',
publicFolder: "public", publicFolder: 'public'
}, },
media: { media: {
tina: { tina: {
mediaRoot: "", mediaRoot: '',
publicFolder: "public", publicFolder: 'public'
}, }
}, },
// See docs on content modeling for more info on how to setup new content models: https://tina.io/docs/schema/ // See docs on content modeling for more info on how to setup new content models: https://tina.io/docs/schema/
schema: { schema: {
collections: [ collections: [
{ {
name: "books", name: 'books',
label: "Library", label: 'Library',
path: 'content/books', path: 'content/books',
match: { match: {
include: "*" include: '*'
}, },
format: "md", format: 'md',
fields: [ fields: [
{ {
type: "string", type: 'string',
name: "title", name: 'title',
label: "Title", label: 'Title',
isTitle: true, isTitle: true,
required: true, required: true
}, },
{ {
type: 'string', type: 'string',
name: "author", name: 'author',
label: "Author", label: 'Author',
required: true required: true
}, },
{ {
type: 'number', type: 'number',
name: "stars", name: 'stars',
label: "Stars", label: 'Stars',
required: true required: true
}, },
{ {
type: 'datetime', type: 'datetime',
name: "readDate", name: 'readDate',
label: "Read date", label: 'Read date',
required: false required: false
}, },
{ {
type: 'string', type: 'string',
name: "url", name: 'url',
label: "URL", label: 'URL',
required: false required: false
}, },
{ {
type: 'string', type: 'string',
name: "thumbnailUrl", name: 'thumbnailUrl',
label: "Thumbnail URL", label: 'Thumbnail URL',
required: false required: false
}, },
{ {
type: 'string', type: 'string',
name: "tags", name: 'tags',
label: "Tags", label: 'Tags',
required: true required: true
}, },
{ {
type: "rich-text", type: 'rich-text',
name: "body", name: 'body',
label: "Body", label: 'Body',
isBody: true, isBody: true
}, }
] ]
}, },
{ {
name: "writing", name: 'writing',
label: "Writing", label: 'Writing',
path: "content/writing", path: 'content/writing',
match: { match: {
include: "*" include: '*'
}, },
format: "md", format: 'md',
fields: [ fields: [
{ {
type: "string", type: 'string',
name: "title", name: 'title',
label: "Title", label: 'Title',
isTitle: true, isTitle: true,
required: true, required: true
}, },
{ {
type: "datetime", type: 'datetime',
name: "pubdate", name: 'pubdate',
label: "Publish Date", label: 'Publish Date'
}, },
{ {
type: "string", type: 'string',
ui: { ui: {
component: "textarea" component: 'textarea'
}, },
name: "desc", name: 'desc',
label: "Description", label: 'Description'
}, },
{ {
type: "string", type: 'string',
name: "tags", name: 'tags',
label: "Tags", label: 'Tags',
list: true list: true
}, },
{ {
type: "rich-text", type: 'rich-text',
name: "body", name: 'body',
label: "Body", label: 'Body',
isBody: true, isBody: true
}, }
], ]
}, }
], ]
}, }
}); })