feat: add tina CMS; layouts
This commit is contained in:
parent
662a461c3c
commit
6fc63fd50c
6 changed files with 79 additions and 72 deletions
|
@ -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",
|
||||||
|
|
|
@ -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>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -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}>
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
122
tina/config.js
122
tina/config.js
|
@ -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
|
||||||
},
|
}
|
||||||
],
|
]
|
||||||
},
|
}
|
||||||
],
|
]
|
||||||
},
|
}
|
||||||
});
|
})
|
||||||
|
|
Loading…
Add table
Reference in a new issue