client: use cookie for theme, redirect post view in server side props

This commit is contained in:
Max Leiter 2022-03-21 14:20:20 -07:00
parent e37bd00a13
commit 1c68aa9765
WARNING! Although there is a key with this ID in the database it does not verify this commit! This commit is SUSPICIOUS.
GPG key ID: A3512F2F2F17EBDA
12 changed files with 641 additions and 236 deletions

View file

@ -20,7 +20,7 @@ const FormattingIcons = ({ textareaRef, setText }: { textareaRef?: RefObject<HTM
// return { textBefore: '', textAfter: '' } // return { textBefore: '', textAfter: '' }
// }, [textareaRef,]) // }, [textareaRef,])
const handleBoldClick = useCallback((e) => { const handleBoldClick = useCallback(() => {
if (textareaRef?.current && setText) { if (textareaRef?.current && setText) {
const selectionStart = textareaRef.current.selectionStart const selectionStart = textareaRef.current.selectionStart
const selectionEnd = textareaRef.current.selectionEnd const selectionEnd = textareaRef.current.selectionEnd
@ -37,7 +37,7 @@ const FormattingIcons = ({ textareaRef, setText }: { textareaRef?: RefObject<HTM
} }
}, [setText, textareaRef]) }, [setText, textareaRef])
const handleItalicClick = useCallback((e) => { const handleItalicClick = useCallback(() => {
if (textareaRef?.current && setText) { if (textareaRef?.current && setText) {
const selectionStart = textareaRef.current.selectionStart const selectionStart = textareaRef.current.selectionStart
const selectionEnd = textareaRef.current.selectionEnd const selectionEnd = textareaRef.current.selectionEnd
@ -52,7 +52,7 @@ const FormattingIcons = ({ textareaRef, setText }: { textareaRef?: RefObject<HTM
} }
}, [setText, textareaRef]) }, [setText, textareaRef])
const handleLinkClick = useCallback((e) => { const handleLinkClick = useCallback(() => {
if (textareaRef?.current && setText) { if (textareaRef?.current && setText) {
const selectionStart = textareaRef.current.selectionStart const selectionStart = textareaRef.current.selectionStart
const selectionEnd = textareaRef.current.selectionEnd const selectionEnd = textareaRef.current.selectionEnd
@ -73,7 +73,7 @@ const FormattingIcons = ({ textareaRef, setText }: { textareaRef?: RefObject<HTM
} }
}, [setText, textareaRef]) }, [setText, textareaRef])
const handleImageClick = useCallback((e) => { const handleImageClick = useCallback(() => {
if (textareaRef?.current && setText) { if (textareaRef?.current && setText) {
const selectionStart = textareaRef.current.selectionStart const selectionStart = textareaRef.current.selectionStart
const selectionEnd = textareaRef.current.selectionEnd const selectionEnd = textareaRef.current.selectionEnd
@ -134,4 +134,4 @@ const FormattingIcons = ({ textareaRef, setText }: { textareaRef?: RefObject<HTM
} }
export default FormattingIcons export default FormattingIcons

View file

@ -5,12 +5,12 @@ import { Select } from '@geist-ui/core'
// import { useAllThemes, useTheme } from '@geist-ui/core' // import { useAllThemes, useTheme } from '@geist-ui/core'
import styles from './header.module.css' import styles from './header.module.css'
import { ThemeProps } from '@lib/types' import { ThemeProps } from '@lib/types'
import Cookies from 'js-cookie'
const Controls = ({ changeTheme, theme }: ThemeProps) => { const Controls = ({ changeTheme, theme }: ThemeProps) => {
const switchThemes = (type: string | string[]) => { const switchThemes = (type: string | string[]) => {
changeTheme() changeTheme()
if (typeof window === 'undefined' || !window.localStorage) return Cookies.set('drift-theme', Array.isArray(type) ? type[0] : type)
window.localStorage.setItem('drift-theme', Array.isArray(type) ? type[0] : type)
} }

View file

@ -1,10 +1,9 @@
import { Button, Text, useTheme, useToasts } from '@geist-ui/core' import { Text, useTheme, useToasts } from '@geist-ui/core'
import { memo, useCallback, useEffect } from 'react' import { memo } from 'react'
import { useDropzone } from 'react-dropzone' import { useDropzone } from 'react-dropzone'
import styles from './drag-and-drop.module.css' import styles from './drag-and-drop.module.css'
import { Document } from '../' import type { Document } from '@lib/types'
import generateUUID from '@lib/generate-uuid' import generateUUID from '@lib/generate-uuid'
import { XCircle } from '@geist-ui/icons'
const allowedFileTypes = [ const allowedFileTypes = [
'application/json', 'application/json',
'application/x-javascript', 'application/x-javascript',
@ -99,7 +98,7 @@ function FileDropzone({ setDocs }: { setDocs: ((docs: Document[]) => void) }) {
const { setToast } = useToasts() const { setToast } = useToasts()
const onDrop = async (acceptedFiles: File[]) => { const onDrop = async (acceptedFiles: File[]) => {
const newDocs = await Promise.all(acceptedFiles.map((file) => { const newDocs = await Promise.all(acceptedFiles.map((file) => {
return new Promise<Document>((resolve, reject) => { return new Promise<Document>((resolve) => {
const reader = new FileReader() const reader = new FileReader()
reader.onabort = () => setToast({ text: 'File reading was aborted', type: 'error' }) reader.onabort = () => setToast({ text: 'File reading was aborted', type: 'error' })

View file

@ -1,4 +1,4 @@
import { Button, ButtonDropdown, Input, Modal, Note, useModal, useToasts } from '@geist-ui/core' import { Button, ButtonDropdown, useToasts } from '@geist-ui/core'
import { useRouter } from 'next/router'; import { useRouter } from 'next/router';
import { useCallback, useState } from 'react' import { useCallback, useState } from 'react'
import generateUUID from '@lib/generate-uuid'; import generateUUID from '@lib/generate-uuid';

View file

@ -1,4 +1,4 @@
import { ChangeEvent, memo } from 'react' import { memo } from 'react'
import { Text, Input } from '@geist-ui/core' import { Text, Input } from '@geist-ui/core'
import ShiftBy from '@components/shift-by' import ShiftBy from '@components/shift-by'
import styles from '../post.module.css' import styles from '../post.module.css'

View file

@ -1,8 +1,8 @@
import { Badge } from "@geist-ui/core" import { Badge } from "@geist-ui/core"
import { Visibility } from "@lib/types" import { PostVisibility } from "@lib/types"
type Props = { type Props = {
visibility: Visibility visibility: PostVisibility
} }
const VisibilityBadge = ({ visibility }: Props) => { const VisibilityBadge = ({ visibility }: Props) => {

View file

@ -6,7 +6,6 @@ const nextConfig = {
reactStrictMode: true, reactStrictMode: true,
experimental: { experimental: {
outputStandalone: true, outputStandalone: true,
optimizeCss: true,
}, },
async rewrites() { async rewrites() {
return [ return [

View file

@ -16,11 +16,11 @@
"@types/js-cookie": "^3.0.1", "@types/js-cookie": "^3.0.1",
"client-zip": "^2.0.0", "client-zip": "^2.0.0",
"comlink": "^4.3.1", "comlink": "^4.3.1",
"critters": "^0.0.16",
"cookie": "^0.4.2", "cookie": "^0.4.2",
"dotenv": "^16.0.0", "dotenv": "^16.0.0",
"js-cookie": "^3.0.1", "js-cookie": "^3.0.1",
"next": "^12.1.1-canary.15", "next": "^12.1.1-canary.15",
"prism-react-renderer": "^1.3.1",
"prismjs": "^1.27.0", "prismjs": "^1.27.0",
"react": "17.0.2", "react": "17.0.2",
"react-debounce-render": "^8.0.2", "react-debounce-render": "^8.0.2",
@ -32,8 +32,8 @@
"react-syntax-highlighter-virtualized-renderer": "^1.1.0", "react-syntax-highlighter-virtualized-renderer": "^1.1.0",
"rehype-autolink-headings": "^6.1.1", "rehype-autolink-headings": "^6.1.1",
"rehype-katex": "^6.0.2", "rehype-katex": "^6.0.2",
"rehype-remark": "^9.1.2",
"rehype-slug": "^5.0.1", "rehype-slug": "^5.0.1",
"rehype-stringify": "^9.0.3",
"remark-gfm": "^3.0.1", "remark-gfm": "^3.0.1",
"remark-math": "^5.1.1", "remark-math": "^5.1.1",
"swr": "^1.2.2" "swr": "^1.2.2"
@ -41,9 +41,11 @@
"devDependencies": { "devDependencies": {
"@types/node": "17.0.21", "@types/node": "17.0.21",
"@types/react": "17.0.39", "@types/react": "17.0.39",
"@types/react-dom": "^17.0.14",
"@types/react-syntax-highlighter": "^13.5.2", "@types/react-syntax-highlighter": "^13.5.2",
"eslint": "8.10.0", "eslint": "8.10.0",
"eslint-config-next": "12.1.0", "eslint-config-next": "12.1.0",
"typescript": "4.6.2" "typescript": "4.6.2",
"typescript-plugin-css-modules": "^3.4.0"
} }
} }

View file

@ -8,12 +8,7 @@ import 'react-loading-skeleton/dist/skeleton.css'
import { SkeletonTheme } from 'react-loading-skeleton'; import { SkeletonTheme } from 'react-loading-skeleton';
import Head from 'next/head'; import Head from 'next/head';
import { ThemeProps } from '@lib/types'; import { ThemeProps } from '@lib/types';
import Cookies from 'js-cookie';
export type PostProps = {
renderedPost: any | null, // Still don't have an official data type for posts
theme: "light" | "dark" | string,
changeTheme: () => void
}
type AppProps<P = any> = { type AppProps<P = any> = {
pageProps: P; pageProps: P;
@ -22,11 +17,10 @@ type AppProps<P = any> = {
export type DriftProps = ThemeProps export type DriftProps = ThemeProps
function MyApp({ Component, pageProps }: AppProps<ThemeProps>) { function MyApp({ Component, pageProps }: AppProps<ThemeProps>) {
const [themeType, setThemeType] = useSharedState<string>('theme', 'light') const [themeType, setThemeType] = useSharedState<string>('theme', Cookies.get('drift-theme') || 'light')
const theme = useTheme();
useEffect(() => { useEffect(() => {
if (typeof window === 'undefined' || !window.localStorage) return const storedTheme = Cookies.get('drift-theme')
const storedTheme = window.localStorage.getItem('drift-theme')
if (storedTheme) setThemeType(storedTheme) if (storedTheme) setThemeType(storedTheme)
// TODO: useReducer? // TODO: useReducer?
}, [setThemeType, themeType]) }, [setThemeType, themeType])

View file

@ -1,45 +1,36 @@
import { Button, Page, Text } from "@geist-ui/core"; import { Button, Page, Text } from "@geist-ui/core";
import Skeleton from 'react-loading-skeleton';
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { useEffect, useState } from "react";
import Document from '../../components/document' import Document from '../../components/document'
import Header from "../../components/header"; import Header from "../../components/header";
import VisibilityBadge from "../../components/visibility-badge"; import VisibilityBadge from "../../components/visibility-badge";
import { PostProps } from "../_app";
import PageSeo from "components/page-seo"; import PageSeo from "components/page-seo";
import styles from './styles.module.css'; import styles from './styles.module.css';
import Cookies from "js-cookie";
import cookie from "cookie"; import cookie from "cookie";
import { GetServerSideProps } from "next"; import { GetServerSideProps } from "next";
import { PostVisibility, ThemeProps } from "@lib/types";
type File = {
id: string
title: string
content: string
}
const Post = ({renderedPost, theme, changeTheme}: PostProps) => { type Files = File[]
const [post, setPost] = useState(renderedPost);
const [isLoading, setIsLoading] = useState(true) export type PostProps = ThemeProps & {
const [error, setError] = useState<string>() post: {
id: string
title: string
description: string
visibility: PostVisibility
files: Files
}
}
const Post = ({ post, theme, changeTheme }: PostProps) => {
const router = useRouter(); const router = useRouter();
useEffect(() => {
async function fetchPost() {
setIsLoading(true);
if (renderedPost) {
setPost(renderedPost)
setIsLoading(false)
return;
}
if (!Cookies.get('drift-token')) {
router.push('/signin');
} else {
setError('Something went wrong fetching the post');
}
}
fetchPost()
}, [router, router.query.id])
const download = async () => { const download = async () => {
const clientZip = require("client-zip") const clientZip = require("client-zip")
@ -59,78 +50,88 @@ const Post = ({renderedPost, theme, changeTheme}: PostProps) => {
return ( return (
<Page width={"100%"}> <Page width={"100%"}>
{!isLoading && ( <PageSeo
<PageSeo title={`${post.title} - Drift`}
title={`${post.title} - Drift`} description={post.description}
description={post.description} isPrivate={post.visibility !== 'public'}
isPrivate={post.visibility === 'private'} />
/>
)}
<Page.Header> <Page.Header>
<Header theme={theme} changeTheme={changeTheme} /> <Header theme={theme} changeTheme={changeTheme} />
</Page.Header> </Page.Header>
<Page.Content width={"var(--main-content-width)"} margin="auto"> <Page.Content width={"var(--main-content-width)"} margin="auto">
{/* {!isLoading && <PostFileExplorer files={post.files} />} */} {/* {!isLoading && <PostFileExplorer files={post.files} />} */}
<div className={styles.header}>
{error && <Text type="error">{error}</Text>} <div className={styles.titleAndBadge}>
{!error && isLoading && <><Text h2><Skeleton width={400} /></Text> <Text h2>{post.title}</Text>
<Document skeleton={true} /> <span><VisibilityBadge visibility={post.visibility} /></span>
</>}
{!isLoading && post && <>
<div className={styles.header}>
<div className={styles.titleAndBadge}>
<Text h2>{post.title}</Text>
<span><VisibilityBadge visibility={post.visibility} /></span>
</div>
<Button auto onClick={download}>
Download as ZIP archive
</Button>
</div> </div>
{post.files.map(({ id, content, title }: { id: any, content: string, title: string }) => ( <Button auto onClick={download}>
<Document Download as ZIP archive
key={id} </Button>
id={id} </div>
content={content} {post.files.map(({ id, content, title }: { id: any, content: string, title: string }) => (
title={title} <Document
editable={false} key={id}
initialTab={'preview'} id={id}
/> content={content}
))} title={title}
</>} editable={false}
initialTab={'preview'}
/>
))}
</Page.Content> </Page.Content>
</Page > </Page >
) )
} }
export const getServerSideProps: GetServerSideProps = async (context) => { export const getServerSideProps: GetServerSideProps = async (context) => {
const headers = context.req.headers
const host = headers.host
const driftToken = cookie.parse(headers.cookie || '')[`drift-token`]
let driftTheme = cookie.parse(headers.cookie || '')[`drift-theme`]
if (driftTheme !== "light" && driftTheme !== "dark") {
driftTheme = "light"
}
const headers = context.req.headers;
const host = headers.host;
const driftToken = cookie.parse(headers.cookie || '')[`drift-token`];
let post;
if (context.query.id) { if (context.query.id) {
post = await fetch('http://' + host + `/server-api/posts/${context.query.id}`, { const post = await fetch('http://' + host + `/server-api/posts/${context.query.id}`, {
method: "GET", method: "GET",
headers: { headers: {
"Content-Type": "application/json", "Content-Type": "application/json",
"Authorization": `Bearer ${driftToken}` "Authorization": `Bearer ${driftToken}`
} }
}); })
if (!post.ok || post.status !== 200) {
return {
redirect: {
destination: '/',
permanent: false,
},
}
}
try { try {
post = await post.json(); const json = await post.json();
const maxAge = 60 * 60 * 24 * 365;
context.res.setHeader(
'Cache-Control',
`${json.visibility === "public" ? "public" : "private"}, s-maxage=${maxAge}, max-age=${maxAge}`
)
return {
props: {
post: json
}
}
} catch (e) { } catch (e) {
console.log(e); console.log(e)
post = null;
} }
} }
return { return {
props: { props: {
renderedPost: post post: null
} }
} }
} }

View file

@ -1,5 +1,6 @@
{ {
"compilerOptions": { "compilerOptions": {
"plugins": [{ "name": "typescript-plugin-css-modules" }],
"target": "es2020", "target": "es2020",
"lib": ["dom", "dom.iterable", "esnext"], "lib": ["dom", "dom.iterable", "esnext"],
"allowJs": true, "allowJs": true,

File diff suppressed because it is too large Load diff