more of the article layout + layout refactor

This commit is contained in:
Tony Sullivan 2023-09-13 09:36:44 -05:00
parent 635fb1510c
commit ae23274489
8 changed files with 386 additions and 291 deletions

View file

Before

Width:  |  Height:  |  Size: 18 KiB

After

Width:  |  Height:  |  Size: 18 KiB

View file

@ -0,0 +1,131 @@
---
import type { HTMLAttributes } from 'astro/types';
import { Image } from 'astro:assets';
import { CollectionEntry } from 'astro:content';
import getReadingTime from 'reading-time';
import FormattedDate from './FormattedDate.astro';
import Icon from './Icon.astro';
import TagList from './TagList.astro';
import settings from '../settings.js';
interface Props extends HTMLAttributes<'header'> {
article: CollectionEntry<'posts'>;
}
const { article, ...attrs } = Astro.props;
const { text: readingTime } = await getReadingTime(article.body);
---
<header {...attrs}>
<a href="/" class="back">
<Icon icon="arrow-left" color="var(--theme-accent-dark)" size="1.25rem" />
<span>Back to feed</span>
</a>
<div class="cover">
{
article.data.cover && (
<Image src={article.data.cover.src} alt={article.data.cover.alt} class="cover" />
)
}
<div class="author">
<Image src={settings.avatar.src} alt={settings.avatar.alt} width={80} />
<p class="u-name">{settings.name}</p>
<p class="u-nickname">{settings.username}</p>
</div>
</div>
<div class="meta">
<FormattedDate date={article.data.pubDate} />
<span>•</span>
<p>
<Icon icon="clock" size="1rem" />
<span>{readingTime}</span>
</p>
</div>
<h1>{article.data.title}</h1>
{article.data.tags?.length > 0 && <TagList tags={article.data.tags} />}
</header>
<style>
header {
display: flex;
flex-direction: column;
gap: 1rem;
}
.back {
text-decoration: none;
font-weight: bold;
}
.back {
display: flex;
align-items: center;
gap: 0.5rem;
margin-inline-start: -4px;
}
.cover {
position: relative;
width: 100%;
min-height: 150px;
background: var(--theme-gradient-main);
border-radius: 1.25rem;
margin-bottom: 1.25rem;
}
.cover > img {
width: 100%;
height: 100%;
aspect-ratio: 2.777;
object-fit: cover;
}
.author {
position: absolute;
display: flex;
align-items: center;
gap: 0.5rem;
bottom: 0%;
right: 3.125rem;
translate: 0 50%;
box-shadow: var(--theme-shadow-md);
background-color: var(--theme-accent-light);
padding: 0.75rem 1.5rem;
border-radius: 1.25rem;
border: 3px solid var(--theme-accent-dark);
}
.author img {
width: 2.5rem;
height: 2.5rem;
}
.author .u-name {
font-family: var(--font-brand);
font-weight: bold;
}
.author .u-nickname {
color: var(--theme-gray-300);
font-size: var(--theme-text-sm);
font-weight: 600;
}
.meta {
display: flex;
align-items: center;
gap: 1rem;
font-size: var(--theme-text-sm);
color: var(--theme-gray-200);
}
.meta svg {
margin-block-end: 0.25rem;
}
</style>

View file

@ -4,28 +4,26 @@ import ThemeToggle from './ThemeToggle.astro';
--- ---
<header> <header>
<a class="site-name" href="/">{settings.username}</a> <a class="site-name" href="/">{settings.username}</a>
<ThemeToggle /> <ThemeToggle />
</header> </header>
<style> <style>
header { header {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
} }
.site-name { .site-name {
font-size: var(--theme-text-lg); font-size: var(--theme-text-lg);
font-weight: 700; font-weight: 700;
font-family: var(--theme-font-brand); font-family: var(--theme-font-brand);
text-decoration: none; text-decoration: none;
color: transparent; color: var(--theme-accent-dark);
background: var(--theme-gradient-main); }
background-clip: text;
}
.site-name:hover { .site-name:hover {
text-decoration: 1px solid underline var(--theme-accent-dark); text-decoration: 1px solid underline var(--theme-accent-dark);
} }
</style> </style>

View file

@ -0,0 +1,33 @@
---
import type { HTMLAttributes } from 'astro/types';
export interface Props extends HTMLAttributes<'article'> {}
---
<article {...Astro.props}>
<slot />
</article>
<style>
article > :global(* + *) {
margin-block-start: 1em;
}
article > :global(p:first-child::first-letter) {
float: left;
font-size: 3.5rem;
padding-inline-end: 0.5rem;
padding-block-start: 0.35rem;
font-weight: bold;
}
article :global(img, video, figure) {
margin-inline: auto;
}
@media (min-width: 50em) {
article > :global(* + *) {
margin-block-start: 1.5em;
}
}
</style>

View file

@ -1,4 +1,5 @@
--- ---
import { Image } from 'astro:assets';
import settings from '../settings'; import settings from '../settings';
import Icon from './Icon.astro'; import Icon from './Icon.astro';
@ -8,155 +9,146 @@ const socialLinks = Object.entries(settings.social) as SocialEntry[];
--- ---
<div class="profile"> <div class="profile">
<div> <div>
<div class="avatar"> <div class="avatar">
<img width="110" height="110" {...settings.avatar} /> <Image {...settings.avatar} width={220} />
</div> </div>
<h1> <h1>
{settings.name} {settings.name}
<small>{settings.username}</small> <small>{settings.username}</small>
</h1> </h1>
</div> </div>
<div class="bio-sections"> <div class="bio-sections">
<div class="bio"> <div class="bio">
<p>🚀 <a href="https://astro.build/">Astro</a> Mascot</p> <p>🚀 <a href="https://astro.build/">Astro</a> Mascot</p>
<p>😊 The cutest</p> <p>😊 The cutest</p>
<p>🎨 Whimsical Speedy Web</p> <p>🎨 Whimsical Speedy Web</p>
</div> </div>
<div class="bio"> <div class="bio">
{ {
settings.pronouns && ( settings.pronouns && (
<p> <p>
<Icon icon="user" color="var(--theme-accent-dark)" size="1.75rem" /> <Icon icon="user" color="var(--theme-accent-dark)" size="1.75rem" />
<span class="sr-only">Pronouns</span> <span class="sr-only">Pronouns</span>
{settings.pronouns} {settings.pronouns}
</p> </p>
) )
} }
{ {
settings.location && ( settings.location && (
<p> <p>
<Icon <Icon icon="location-point" color="var(--theme-accent-dark)" size="1.75rem" />
icon="location-point" <span class="sr-only">Location</span>
color="var(--theme-accent-dark)" {settings.location}
size="1.75rem" </p>
/> )
<span class="sr-only">Location</span> }
{settings.location} {
</p> settings.homepage && (
) <p>
} <Icon icon="link-h" color="var(--theme-accent-dark)" size="1.75rem" />
{ <span class="sr-only">Homepage</span>
settings.homepage && ( <a href={settings.homepage}>{settings.homepage.replace(/^https?:\/\/(www\.)?/, '')}</a>
<p> </p>
<Icon )
icon="link-h" }
color="var(--theme-accent-dark)" <p>
size="1.75rem" <Icon icon="rss-alt" color="var(--theme-accent-dark)" size="1.75rem" />
/> <a href="/rss.xml">RSS Feed</a>
<span class="sr-only">Homepage</span> </p>
<a href={settings.homepage}> </div>
{settings.homepage.replace(/^https?:\/\/(www\.)?/, '')} <ul class="social">
</a> {
</p> socialLinks.map(([key, url]) => (
) <li>
} <a href={url}>
<p> <Icon icon={`${key}-logo`} size="1.75rem" />
<Icon icon="rss-alt" color="var(--theme-accent-dark)" size="1.75rem" /> </a>
<a href="/rss.xml">RSS Feed</a> </li>
</p> ))
</div> }
<ul class="social"> </ul>
{ </div>
socialLinks.map(([key, url]) => (
<li>
<a href={url}>
<Icon icon={`${key}-logo`} size="1.75rem" />
</a>
</li>
))
}
</ul>
</div>
</div> </div>
<style> <style>
.profile { .profile {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 1.5rem; gap: 1.5rem;
} }
.avatar { .avatar {
display: inline-block; display: inline-block;
position: relative; position: relative;
} }
.avatar::after { .avatar::after {
border-radius: var(--theme-radius-full); border-radius: var(--theme-radius-full);
position: absolute; position: absolute;
content: ''; content: '';
inset: 0; inset: 0;
border: 0.1875rem solid var(--theme-text); border: 0.1875rem solid var(--theme-text);
} }
.avatar img { .avatar img {
height: auto; width: 110px;
} height: 110px;
}
h1 { h1 {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
font-size: var(--theme-text-xl); font-size: var(--theme-text-xl);
} }
small { small {
font-size: var(--theme-text-base); font-size: var(--theme-text-base);
font-family: var(--theme-font-body); font-family: var(--theme-font-body);
color: var(--theme-gray-200); color: var(--theme-gray-200);
} }
.bio-sections { .bio-sections {
display: grid; display: grid;
grid-template-columns: 1fr 1fr; grid-template-columns: 1fr 1fr;
align-items: flex-start; align-items: flex-start;
gap: 1rem 1.75rem; gap: 1rem 1.75rem;
} }
.bio-sections > :nth-child(2) { .bio-sections > :nth-child(2) {
grid-row: span 2; grid-row: span 2;
} }
.bio { .bio {
font-size: var(--theme-text-sm); font-size: var(--theme-text-sm);
font-weight: 500; font-weight: 500;
} }
.bio > * + * { .bio > * + * {
margin-top: 0.75rem; margin-top: 0.75rem;
} }
.bio :global(svg) { .bio :global(svg) {
/* Slightly hacky way to avoid the icon height being included in the box calculation. */ /* Slightly hacky way to avoid the icon height being included in the box calculation. */
margin: -50% 0; margin: -50% 0;
} }
.social { .social {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5rem; gap: 0.5rem;
list-style: none; list-style: none;
padding: 0; padding: 0;
} }
.social a { .social a {
color: var(--theme-accent-dark); color: var(--theme-accent-dark);
} }
@media (min-width: 50em) { @media (min-width: 50em) {
.bio-sections { .bio-sections {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
} }
} }
</style> </style>

View file

@ -1,11 +1,9 @@
--- ---
import { Image } from 'astro:assets';
import type { CollectionEntry } from 'astro:content'; import type { CollectionEntry } from 'astro:content';
import getReadingTime from 'reading-time';
import Layout, { type Props as LayoutProps } from './Base.astro'; import Layout, { type Props as LayoutProps } from './Base.astro';
import FormattedDate from '../components/FormattedDate.astro'; import ArticleHeader from '../components/ArticleHeader.astro';
import TagList from '../components/TagList.astro';
import Icon from '../components/Icon.astro'; import Icon from '../components/Icon.astro';
import Prose from '../components/Prose.astro';
export interface Props extends LayoutProps { export interface Props extends LayoutProps {
article: CollectionEntry<'posts'>; article: CollectionEntry<'posts'>;
@ -16,118 +14,56 @@ export interface Props extends LayoutProps {
const { article, next, prev } = Astro.props; const { article, next, prev } = Astro.props;
const { Content, headings } = await article.render(); const { Content, headings } = await article.render();
const readingTime = getReadingTime(article.body).text;
--- ---
<Layout ...Astro.props} wrapperReverse> <Layout ...Astro.props} wrapperReverse>
<header> <ArticleHeader {article} class="header" />
<a href="/" class="back">
<Icon icon="arrow-left" color="var(--theme-accent-dark)" size="1.25rem" /> <Prose class="content">
<span>Back to feed</span>
</a>
{article.data.cover && <Image src={article.data.cover.src} alt={article.data.cover.alt} class="cover" />}
<div class="meta">
<FormattedDate date={article.data.pubDate} />
<span>•</span>
<p>
<Icon icon="clock" size="1rem" />
<span>{readingTime}</span>
</p>
</div>
<h1>{article.data.title}</h1>
{article.data.tags?.length > 0 && (
<TagList tags={article.data.tags} />
)}
</header>
<div class="content">
<Content /> <Content />
</div> </Prose>
{headings.length > 0 && (
<aside> {
<h3>Table of contents</h3> headings.length > 0 && (
<ul> <aside>
{headings.map(({ slug, text }) => ( <h3>Table of contents</h3>
<li> <ul>
<a href={`${Astro.url.pathname}#${slug}`}>{text}</a> {headings.map(({ slug, text }) => (
</li> <li>
))} <a href={`${Astro.url.pathname}#${slug}`}>{text}</a>
</ul> </li>
</aside> ))}
)} </ul>
{(next || prev) && ( </aside>
<footer> )
{prev && <a href={`/post/${prev.slug}`} class="prev"> }
<Icon icon="arrow-left" size="1.25rem" color="var(--theme-accent-dark)" />
<span>Previous: {prev.data.title}</span> {
</a>} (next || prev) && (
{next && <a href={`/post/${next.slug}`} class="next"> <footer>
<span>Next: {next.data.title}</span> {prev && (
<Icon icon="arrow-right" size="1.25rem" color="var(--theme-accent-dark)" /> <a href={`/post/${prev.slug}`} class="prev">
</a>} <Icon icon="arrow-left" size="1.25rem" color="var(--theme-accent-dark)" />
</footer> <span>Previous: {prev.data.title}</span>
)} </a>
)}
{next && (
<a href={`/post/${next.slug}`} class="next">
<span>Next: {next.data.title}</span>
<Icon icon="arrow-right" size="1.25rem" color="var(--theme-accent-dark)" />
</a>
)}
</footer>
)
}
</Layout> </Layout>
<style> <style>
header, footer { .header,
footer {
grid-column: 1 / -1; grid-column: 1 / -1;
} }
header {
display: flex;
flex-direction: column;
gap: 1rem;
}
.content > :global(* + *) {
margin-block-start: 1em;
}
.content > :global(p:first-child::first-letter) {
float: left;
font-size: 3.5rem;
padding-inline-end: .5rem;
padding-block-start: .35rem;
font-weight: bold;
}
.content :global(img, video, figure) {
margin-inline: auto;
}
.meta {
display: flex;
align-items: center;
gap: 1rem;
font-size: var(--theme-text-sm);
color: var(--theme-gray-200);
}
.meta svg {
margin-block-end: 0.25rem;
}
.back {
text-decoration: none;
font-weight: bold;
}
.back {
display: flex;
align-items: center;
gap: 0.5rem;
margin-inline-start: -4px;
}
.cover {
height: auto;
aspect-ratio: 2.777;
object-fit: cover;
border-radius: 1.25rem;
margin-block-start: 0.5rem;
}
footer { footer {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
@ -165,9 +101,5 @@ const readingTime = getReadingTime(article.body).text;
aside { aside {
display: block; display: block;
} }
.content > :global(* + *) {
margin-block-start: 1.5em;
}
} }
</style> </style>

View file

@ -1,5 +1,6 @@
--- ---
import type { GetStaticPathsOptions, Page } from 'astro'; import type { GetStaticPathsOptions, Page } from 'astro';
import { Image } from 'astro:assets';
import type { CollectionEntry } from 'astro:content'; import type { CollectionEntry } from 'astro:content';
import Pagination from '../components/Pagination.astro'; import Pagination from '../components/Pagination.astro';
import TagList from '../components/TagList.astro'; import TagList from '../components/TagList.astro';
@ -28,11 +29,11 @@ const { page } = Astro.props;
<article class="card"> <article class="card">
<header> <header>
<div style="display: flex;gap:0.875rem;align-items:center;"> <div style="display: flex;gap:0.875rem;align-items:center;">
<img <Image
style="border-radius: var(--theme-radius-full);background-color:var(--theme-shade-subtle);" src={settings.avatar.src}
width="60" width={120}
height="60" class="u-avatar"
{...settings.avatar} alt={settings.avatar.alt}
/> />
<div> <div>
<p style="font-family:var(--theme-font-brand);font-weight:700;font-size:var(--theme-text-lg)"> <p style="font-family:var(--theme-font-brand);font-weight:700;font-size:var(--theme-text-lg)">
@ -89,4 +90,10 @@ const { page } = Astro.props;
header > * + * { header > * + * {
margin-top: 0.625rem; margin-top: 0.625rem;
} }
.u-avatar {
width: 3.75rem;
height: 3.75rem;
border-radius: var(--theme-radius-full);
background-color: var(--theme-shade-subtle);
}
</style> </style>

View file

@ -1,24 +1,26 @@
import avatar from './assets/avatar.webp'
export default { export default {
name: 'Houston Astro', name: 'Houston Astro',
username: '@houston', username: '@houston',
avatar: { avatar: {
src: '/avatar.webp', src: avatar,
alt: 'Astro mascot Houston smiling', alt: 'Astro mascot Houston smiling',
}, },
rss: { rss: {
title: 'Houston Astros Feed', title: 'Houston Astros Feed',
description: 'Stay up-to-date with the latest posts from Houston Astro!', description: 'Stay up-to-date with the latest posts from Houston Astro!',
}, },
pronouns: 'They/Them', pronouns: 'They/Them',
location: 'Space', location: 'Space',
homepage: 'https://astro.build', homepage: 'https://astro.build',
social: { social: {
twitter: 'https://twitter.com/astrodotbuild', twitter: 'https://twitter.com/astrodotbuild',
twitch: 'https://www.twitch.tv/bholmesdev', twitch: 'https://www.twitch.tv/bholmesdev',
github: 'https://github.com/withastro', github: 'https://github.com/withastro',
devto: 'https://dev.to/search?q=astro', devto: 'https://dev.to/search?q=astro',
codepen: 'https://codepen.io/delucis', codepen: 'https://codepen.io/delucis',
mastodon: 'https://m.webtoo.ls/@astro', mastodon: 'https://m.webtoo.ls/@astro',
youtube: 'https://www.youtube.com/@astrodotbuild', youtube: 'https://www.youtube.com/@astrodotbuild',
}, },
} as const; } as const;