feat: layout tweaks; add books
This commit is contained in:
parent
6fc63fd50c
commit
09bd28f64b
18 changed files with 132 additions and 21 deletions
10
content/books/1984.md
Normal file
10
content/books/1984.md
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
---
|
||||||
|
title: '1984'
|
||||||
|
author: George Orwell
|
||||||
|
stars: 3
|
||||||
|
readDate: 2023-07-31T23:00:00.000Z
|
||||||
|
url: 'https://app.thestorygraph.com/books/6ff3f487-8d37-4ac5-8190-6622d6562639'
|
||||||
|
thumbnailUrl: 'https://cdn.thestorygraph.com/v43bj24inkwioiogb5uz8nqmpnpw'
|
||||||
|
tags: 'fiction, dystopian, classics'
|
||||||
|
---
|
||||||
|
|
10
content/books/A-Monster-Calls.md
Normal file
10
content/books/A-Monster-Calls.md
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
---
|
||||||
|
title: A Monster Calls
|
||||||
|
author: Patrick Ness
|
||||||
|
stars: 4
|
||||||
|
readDate: 2024-05-31T23:00:00.000Z
|
||||||
|
url: 'https://app.thestorygraph.com/books/170c1204-1410-4246-babb-c80e31aebea9'
|
||||||
|
thumbnailUrl: 'https://cdn.thestorygraph.com/50qkuazqx2lidfd0hk74ltifz9nf'
|
||||||
|
tags: 'fiction, young adult'
|
||||||
|
---
|
||||||
|
|
10
content/books/Diary-of-An-Oxygen-Thief.md
Normal file
10
content/books/Diary-of-An-Oxygen-Thief.md
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
---
|
||||||
|
title: Diary of An Oxygen Thief
|
||||||
|
author: Anonymous
|
||||||
|
stars: 4
|
||||||
|
readDate: 2024-03-01T00:00:00.000Z
|
||||||
|
url: 'https://app.thestorygraph.com/books/9ce581f2-d9ac-4be1-abf7-4597684bab7f'
|
||||||
|
thumbnailUrl: 'https://cdn.thestorygraph.com/l5zkpt8v2wj76ri6nkq7ismqsskl'
|
||||||
|
tags: 'romance, fiction'
|
||||||
|
---
|
||||||
|
|
10
content/books/No-God-But-God.md
Normal file
10
content/books/No-God-But-God.md
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
---
|
||||||
|
title: No God But God
|
||||||
|
author: Reza Aslan
|
||||||
|
stars: 4
|
||||||
|
readDate: 2023-01-01T00:00:00.000Z
|
||||||
|
url: 'https://app.thestorygraph.com/books/27cb3f11-77c6-4eca-9344-d64885e6f1c4'
|
||||||
|
thumbnailUrl: 'https://cdn.thestorygraph.com/gf3c92roqrgdbdsgphns5n111t93'
|
||||||
|
tags: 'non-fiction, religion, history'
|
||||||
|
---
|
||||||
|
|
10
content/books/The-Nature-of-Alexander.md
Normal file
10
content/books/The-Nature-of-Alexander.md
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
---
|
||||||
|
title: The Nature of Alexander
|
||||||
|
author: Mary Renault
|
||||||
|
stars: 4.5
|
||||||
|
readDate: 2023-03-31T23:00:00.000Z
|
||||||
|
url: 'https://app.thestorygraph.com/books/bc111e8b-1b3f-466a-9efd-c3316ae4b533'
|
||||||
|
thumbnailUrl: 'https://cdn.thestorygraph.com/68zjkhp67u2bi6qh3melbyqaly06'
|
||||||
|
tags: 'non-fiction, history, biography'
|
||||||
|
---
|
||||||
|
|
|
@ -1,12 +1,17 @@
|
||||||
import { formatDate } from '@/lib/helpers'
|
import { formatDate } from '@/lib/helpers'
|
||||||
import { NextSeo } from 'next-seo'
|
import { NextSeo } from 'next-seo'
|
||||||
|
import Image from 'next/image'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
|
import style from "./Book.module.css";
|
||||||
|
import ExternalLink from '../ExternalLink/ExternalLink'
|
||||||
|
|
||||||
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>
|
||||||
|
<Link href='./'>Back...</Link>
|
||||||
|
|
||||||
<article>
|
<article>
|
||||||
<NextSeo
|
<NextSeo
|
||||||
title={attributes.title} description={attributes.desc} openGraph={
|
title={attributes.title} description={attributes.desc} openGraph={
|
||||||
|
@ -20,10 +25,19 @@ function Book ({ attributes, html }) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<Link href='./'>Back...</Link>
|
<div className={style.layout}>
|
||||||
{attributes.pubdate && <p>{formatDate(attributes.pubdate)}</p>}
|
<Image src={attributes.thumbnailUrl} width={250} height={580} alt='' className={style.thumbnail} />
|
||||||
<div data-test='content' dangerouslySetInnerHTML={{ __html: html || '<p>(no review)</p>' }} />
|
<div>
|
||||||
|
<div data-test='content' dangerouslySetInnerHTML={{ __html: html || '<p>(no review)</p>' }} />
|
||||||
|
<p>
|
||||||
|
<span className='bold'>Rating:</span> {attributes.stars}/5<br />
|
||||||
|
<span className='bold'>Read on:</span> {formatDate(attributes.readDate)}
|
||||||
|
</p>
|
||||||
|
<p><ExternalLink href={attributes.url}>View on The StoryGraph</ExternalLink></p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</article>
|
</article>
|
||||||
</>
|
</>
|
||||||
|
|
34
src/components/Book/Book.module.css
Normal file
34
src/components/Book/Book.module.css
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
|
||||||
|
.layout {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
gap: 1.25rem;
|
||||||
|
margin: 1.25rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layout:first-child {
|
||||||
|
flex-grow: 0;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layout:last-child {
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layout p:first-child {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.thumbnail {
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 650px) {
|
||||||
|
.layout {
|
||||||
|
flex-direction: column-reverse;
|
||||||
|
}
|
||||||
|
|
||||||
|
.thumbnail {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
|
@ -17,7 +17,7 @@ export default function BookListItem ({ href, title, author, stars, readDate, ur
|
||||||
</h2>
|
</h2>
|
||||||
<p className={style.author}>{author}</p>
|
<p className={style.author}>{author}</p>
|
||||||
<p>{tags}</p>
|
<p>{tags}</p>
|
||||||
<p>{stars}/5</p>
|
{/* <p>{stars}/5</p> */}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
|
|
||||||
.thumb {
|
.thumb {
|
||||||
width: auto;
|
width: auto;
|
||||||
height: 200px;
|
height: 290px;
|
||||||
background-position: center;
|
background-position: center;
|
||||||
background-repeat: no-repeat;
|
background-repeat: no-repeat;
|
||||||
background-size: cover;
|
background-size: cover;
|
||||||
|
|
|
@ -3,9 +3,7 @@ import style from './Grid.module.css'
|
||||||
export default function Grid ({ columns, children }) {
|
export default function Grid ({ columns, children }) {
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={style.grid} style={{
|
className={style.grid}
|
||||||
gridTemplateColumns: `repeat(${columns}, 1fr)`
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
.grid {
|
.grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 25px;
|
grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
|
||||||
}
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import { formatDate } from '@/lib/helpers'
|
import { formatDate } from '@/lib/helpers'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
|
|
||||||
export default function StaticContentList ({ entries, urlPrefix }) {
|
export default function StaticContentList ({ entries, urlPrefix, max = 0 }) {
|
||||||
return (
|
return (
|
||||||
<table>
|
<table>
|
||||||
<tbody>
|
<tbody>
|
||||||
|
@ -12,7 +12,7 @@ export default function StaticContentList ({ entries, urlPrefix }) {
|
||||||
<Link href={`${urlPrefix}${e.slug}`}>{e.attributes.title}</Link>
|
<Link href={`${urlPrefix}${e.slug}`}>{e.attributes.title}</Link>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
))}
|
)).slice(0, max > 0 ? max : entries.length)}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
)
|
)
|
||||||
|
|
|
@ -66,8 +66,7 @@ export function getContentTags (contentPath) {
|
||||||
for (const entry of entries) {
|
for (const entry of entries) {
|
||||||
if (!entry.attributes.tags) { continue }
|
if (!entry.attributes.tags) { continue }
|
||||||
|
|
||||||
const tags = entry.attributes.tags.split(', ')
|
const tags = entry.attributes.tags;
|
||||||
|
|
||||||
for (const tag of tags) {
|
for (const tag of tags) {
|
||||||
allTags[tag] = !allTags[tag] ? 1 : allTags[tag] + 1
|
allTags[tag] = !allTags[tag] ? 1 : allTags[tag] + 1
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,3 +7,13 @@ export function toSlug (input) {
|
||||||
export function formatDate (date) {
|
export function formatDate (date) {
|
||||||
return dateFns.format(Date.parse(date), 'PPP')
|
return dateFns.format(Date.parse(date), 'PPP')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Silliness to make sure dates don't get passed to the
|
||||||
|
* page function below as [object Object]
|
||||||
|
* @param {*} obj
|
||||||
|
* @returns
|
||||||
|
*/
|
||||||
|
export function stringifyAndParse(obj) {
|
||||||
|
return JSON.parse(JSON.stringify(obj));
|
||||||
|
}
|
|
@ -33,7 +33,7 @@ export default function Home ({ postEntries }) {
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<h2>Recent posts</h2>
|
<h2>Recent posts</h2>
|
||||||
<StaticContentList entries={postEntries} urlPrefix='writing/' />
|
<StaticContentList entries={postEntries} urlPrefix='writing/' max={5}/>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
</DefaultLayout>
|
</DefaultLayout>
|
||||||
|
|
|
@ -2,9 +2,11 @@ import React from 'react'
|
||||||
import DefaultLayout from '@/layouts/DefaultLayout/DefaultLayout'
|
import DefaultLayout from '@/layouts/DefaultLayout/DefaultLayout'
|
||||||
import { getStaticEntryPaths, getStaticEntryProps } from '@/lib/content'
|
import { getStaticEntryPaths, getStaticEntryProps } from '@/lib/content'
|
||||||
import Book from '@/components/Book/Book'
|
import Book from '@/components/Book/Book'
|
||||||
|
import { stringifyAndParse } from '@/lib/helpers'
|
||||||
|
|
||||||
export const getStaticPaths = () => getStaticEntryPaths('./content/books')
|
export const getStaticPaths = () => getStaticEntryPaths('./content/books')
|
||||||
export const getStaticProps = (ctx) => getStaticEntryProps('./content/books', ctx)
|
export const getStaticProps = (ctx) =>
|
||||||
|
stringifyAndParse(getStaticEntryProps('./content/books', ctx));
|
||||||
|
|
||||||
export default function LibrarySingle ({ attributes, html }) {
|
export default function LibrarySingle ({ attributes, html }) {
|
||||||
return (
|
return (
|
||||||
|
|
|
@ -2,6 +2,7 @@ import BookListItem from '@/components/BookListItem/BookListItem'
|
||||||
import Grid from '@/components/Grid/Grid'
|
import Grid from '@/components/Grid/Grid'
|
||||||
import DefaultLayout from '@/layouts/DefaultLayout/DefaultLayout'
|
import DefaultLayout from '@/layouts/DefaultLayout/DefaultLayout'
|
||||||
import { getStaticEntries } from '@/lib/content'
|
import { getStaticEntries } from '@/lib/content'
|
||||||
|
import { stringifyAndParse } from '@/lib/helpers'
|
||||||
import { NextSeo } from 'next-seo'
|
import { NextSeo } from 'next-seo'
|
||||||
|
|
||||||
export const Title = 'Library'
|
export const Title = 'Library'
|
||||||
|
@ -14,11 +15,9 @@ export const getStaticProps = () => {
|
||||||
|
|
||||||
return {
|
return {
|
||||||
props: {
|
props: {
|
||||||
// Silliness to make sure dates don't get passed to the
|
bookEntries: stringifyAndParse(bookEntries)
|
||||||
// page function below as [object Object]
|
|
||||||
bookEntries: JSON.parse(JSON.stringify(bookEntries))
|
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Library ({ bookEntries }) {
|
export default function Library ({ bookEntries }) {
|
||||||
|
|
|
@ -17,4 +17,8 @@ td:first-child {
|
||||||
|
|
||||||
.form-group {
|
.form-group {
|
||||||
margin: 20px 0;
|
margin: 20px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bold {
|
||||||
|
font-weight: 700;
|
||||||
}
|
}
|
Loading…
Add table
Reference in a new issue