You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

232 lines
4.9 KiB

import React, { useEffect, useState } from "react";
import { useLocation } from "react-router-dom";
import { marked } from "marked";
import { Article, blog, getBlogPath, nextAndPrev } from "../../../data";
import "../../../blog.css";
import { ArticleSubHeader } from "./ArticleSubHeader";
import { css, cx } from "@emotion/css";
import { ReactComponent as Arrow } from "../../../assets/arrow-thin.svg";
import { ReactComponent as Close } from "../../../assets/close.svg";
import { ellipses, useNav } from "../../../util";
const Markdown: React.FC<{ content: string }> = ({ content }) => {
return (
<div
className={css`
& > * {
width: 100%;
}
& img {
width: 100%;
padding-block: 1.5rem;
}
& blockquote {
margin-inline: 0;
padding-inline-start: 1.5rem;
border-inline-start: 1px solid var(--text-colour);
}
`}
dangerouslySetInnerHTML={{ __html: marked(content) }}></div>
);
};
const Preview: React.FC<{ article: Article }> = ({ article }) => {
return (
<div
className={cx(
"preview",
css`
background-color: #353535;
opacity: 0;
pointer-events: none;
position: absolute;
transition: opacity 300ms;
max-width: 100%;
bottom: 4rem;
right: 0;
border-radius: 0.5rem;
overflow: hidden;
width: 13rem;
display: flex;
flex-direction: column;
& * {
line-height: 1.25em;
}
& img {
max-width: 100%;
}
& header {
padding: 0.6rem 1rem 0.8rem 1rem;
display: flex;
flex-direction: column;
gap: 0.5rem;
}
& h3 {
font-size: 1.2rem;
}
& p {
font-size: 0.9rem;
color: var(--text-subdued);
}
`,
)}>
<img src={"/blog/assets/" + article["featured-img"]} alt="Featured" />
<header>
<h3>{article.title}</h3>
<p>{ellipses(article.snippet, 110)}</p>
</header>
</div>
);
};
const btn = css`
background-color: #535353;
border: 0;
cursor: pointer;
border-radius: 0.5rem;
padding: 0.5rem 1.4rem;
color: #c8c8c8;
font-size: 1.2rem;
display: flex;
align-items: center;
gap: 0.8rem;
font-weight: 600;
transition: background-color 150ms;
&:hover {
background-color: #414141;
color: var(--text-colour);
}
& svg {
height: 2rem;
width: 1.5rem;
}
`;
export const BlogPost: React.FC = () => {
const navigate = useNav();
const location = useLocation();
const [content, setContent] = useState("");
const [error, setError] = useState("");
const [loading, setLoading] = useState(true);
const [year, slug] = location.pathname.split("/").slice(-2);
const article = blog[year]?.[slug];
const [next, prev] = nextAndPrev(year, slug);
useEffect(() => {
async function query() {
setLoading(true);
const path = getBlogPath(article) + ".md";
const res = await fetch(path);
console.log(res.status);
// not success and not a cached response
if (res.status > 299 && res.status !== 304) {
if (res.status > 399) {
const err = await res.text().catch(() => "Unknown error");
return setError(err);
} else return setError("Unexpected redirect");
}
const content = await res.text();
setContent(content.split("---").slice(2).join("---"));
setLoading(false);
}
query();
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [location]);
if (loading) return <div>Loading...</div>;
if (!article || error) return <div>{error || "Unknown error occurred"}</div>;
return (
<>
<Close
role="button"
onClick={navigate("/blog")}
className={css`
min-height: 1rem;
min-width: 1rem;
cursor: pointer;
align-self: flex-end;
`}
/>
<div
className={css`
width: 100%;
max-height: 25rem;
`}>
<img
className={css`
max-width: 100%;
height: 100%;
border-radius: 0.5rem;
`}
src={"/blog/assets/" + article["featured-img"]}
alt="Featured"
/>
</div>
<h1
className={css`
font-size: 2.2rem;
`}>
{article.title}
</h1>
<ArticleSubHeader article={article} />
<Markdown content={content} />
<div
className={css`
display: inline-flex;
justify-content: flex-end;
gap: 1rem;
position: relative;
& .btn-holder:hover .preview {
opacity: 100;
pointer-events: all;
}
`}>
{prev && (
<span className="btn-holder">
<Preview article={prev} />
<button className={btn} onClick={navigate(getBlogPath(prev))}>
<Arrow
className={css`
transform: rotate(180deg);
`}
/>
{!next ? <span>Previous</span> : ""}
</button>
</span>
)}
{next && (
<span className="btn-holder">
<Preview article={next} />
<button className={btn} onClick={navigate(getBlogPath(next))}>
<span
className={css`
padding-bottom: 0.1em;
`}>
Next
</span>
<Arrow />
</button>
</span>
)}
</div>
</>
);
};