diff --git a/src/app/notes/Paper.tsx b/src/app/notes/Paper.tsx index f6cea61f29..d060c956ea 100644 --- a/src/app/notes/Paper.tsx +++ b/src/app/notes/Paper.tsx @@ -6,7 +6,7 @@ export const Paper: Component = ({ children }) => { className={clsx( 'relative bg-base-100 md:col-start-1 lg:col-auto', '-m-4 p-[2rem_1rem] lg:m-0 lg:p-[30px_45px]', - 'rounded-[0_6px_6px_0] border border-[#bbb3]', + 'rounded-[0_6px_6px_0] border border-[#bbb3] shadow-sm dark:shadow-[#333]', 'note-layout-main', )} > diff --git a/src/app/notes/[id]/page.module.css b/src/app/notes/[id]/page.module.css new file mode 100644 index 0000000000..c485b6a2d9 --- /dev/null +++ b/src/app/notes/[id]/page.module.css @@ -0,0 +1,36 @@ +.with-indent { + :global { + ul .indent, + .paragraph .indent { + border-bottom: 1px solid #00b8bb41; + } + + .paragraph:not(:nth-child(1)) > span.indent { + &:nth-child(1) { + margin-left: 2rem; + } + } + + main { + > p:first-child { + margin-bottom: 2rem; + } + + .paragraph:first-child::first-letter { + float: left; + font-size: 2.4em; + margin: 0 0.2em 0 0; + } + } + } +} + +.with-serif { + :global { + main { + @apply font-serif; + + font-size: 16px; + } + } +} diff --git a/src/app/notes/[id]/page.tsx b/src/app/notes/[id]/page.tsx index f9fc168593..4a4ebb1857 100644 --- a/src/app/notes/[id]/page.tsx +++ b/src/app/notes/[id]/page.tsx @@ -2,8 +2,10 @@ import { useEffect } from 'react' import { Balancer } from 'react-wrap-balancer' +import clsx from 'clsx' import dayjs from 'dayjs' import { useParams } from 'next/navigation' +import type { MarkdownToJSX } from '~/components/ui/markdown' import { PageDataHolder } from '~/components/common/PageHolder' import { useSetHeaderMetaInfo } from '~/components/layout/header/internal/hooks' @@ -16,6 +18,8 @@ import { ArticleElementProvider } from '~/providers/article/article-element-prov import { useSetCurrentNoteId } from '~/providers/note/current-note-id-provider' import { NoteLayoutRightSidePortal } from '~/providers/note/right-side-provider' +import styles from './page.module.css' + const PageImpl = () => { const { id } = useParams() as { id: string } const { data } = useNoteByNidQuery(id) @@ -52,7 +56,9 @@ const PageImpl = () => { .format('YYYY 年 M 月 D 日 dddd') return ( -
+

{note.title} @@ -66,7 +72,7 @@ const PageImpl = () => {

- + @@ -77,6 +83,17 @@ const PageImpl = () => { ) } +const Markdownrenderers: { [name: string]: Partial } = { + text: { + react(node, _, state) { + return ( + + {node.content} + + ) + }, + }, +} export default PageDataHolder(PageImpl, () => { const { id } = useParams() as { id: string } return useNoteByNidQuery(id) diff --git a/src/components/common/Lazyload.tsx b/src/components/common/Lazyload.tsx new file mode 100644 index 0000000000..b82e090696 --- /dev/null +++ b/src/components/common/Lazyload.tsx @@ -0,0 +1,23 @@ +import React from 'react' +import { useInView } from 'react-intersection-observer' +import type { FC, PropsWithChildren } from 'react' +import type { IntersectionOptions } from 'react-intersection-observer' + +export type LazyLoadProps = { + offset?: number + placeholder?: React.ReactNode +} & IntersectionOptions +export const LazyLoad: FC = (props) => { + const { placeholder = null, offset = 0, ...rest } = props + const { ref, inView } = useInView({ + triggerOnce: true, + rootMargin: `${offset || 0}px`, + ...rest, + }) + return ( + <> + + {!inView ? placeholder : props.children} + + ) +} diff --git a/src/components/ui/image/LazyImage.tsx b/src/components/ui/image/LazyImage.tsx new file mode 100644 index 0000000000..2b8c49469e --- /dev/null +++ b/src/components/ui/image/LazyImage.tsx @@ -0,0 +1,271 @@ +import React, { + forwardRef, + memo, + useCallback, + useEffect, + useImperativeHandle, + useMemo, + useRef, + useState, +} from 'react' +import { clsx } from 'clsx' +import mediumZoom from 'medium-zoom' +import type { + CSSProperties, + DetailedHTMLProps, + FC, + ImgHTMLAttributes, +} from 'react' + +import styles from './index.module.css' +import { useCalculateNaturalSize } from './use-calculate-size' + +interface ImageProps { + defaultImage?: string + src: string + alt?: string + height?: number | string + width?: number | string + backgroundColor?: string + popup?: boolean + overflowHidden?: boolean + getParentElWidth?: ((parentElementWidth: number) => number) | number + showErrorMessage?: boolean +} + +const Image: FC< + { + popup?: boolean + height?: number | string + width?: number | string + loaderFn: () => void + loaded: boolean + } & Pick< + DetailedHTMLProps, HTMLImageElement>, + 'src' | 'alt' + > +> = memo(({ src, alt, height, width, popup = false, loaded, loaderFn }) => { + const imageRef = useRef(null) + + useEffect(() => { + if (!popup) { + return + } + const $image = imageRef.current + if ($image) { + const zoom = mediumZoom($image, { + background: 'var(--light-bg)', + }) + + return () => { + zoom.detach(zoom.getImages()) + } + } + }, [popup]) + + useEffect(() => { + loaderFn() + }, [loaderFn]) + + return ( + <> +
+ {alt} +
+ + ) +}) + +const onImageAnimationEnd: React.AnimationEventHandler = ( + e, +) => { + ;(e.target as HTMLElement).dataset.animated = '1' +} + +export type ImageLazyRef = { status: 'loading' | 'loaded' } + +export const LazyImage = memo( + forwardRef< + ImageLazyRef, + ImageProps & + DetailedHTMLProps, HTMLImageElement> + >((props, ref) => { + const { + defaultImage, + src, + alt, + height, + width, + backgroundColor = 'rgb(111,111,111)', + popup = false, + style, + overflowHidden = false, + getParentElWidth = (w) => w, + showErrorMessage, + ...rest + } = props + useImperativeHandle(ref, () => { + return { + status: loaded ? 'loaded' : ('loading' as any), + } + }) + const realImageRef = useRef(null) + const placeholderRef = useRef(null) + + const wrapRef = useRef(null) + const [calculatedSize, calculateDimensions] = useCalculateNaturalSize() + + const [loaded, setLoad] = useState(false) + const loaderFn = useCallback(() => { + if (!src || loaded) { + return + } + + const image = new window.Image() + image.src = src as string + // FIXME + const parentElement = wrapRef.current?.parentElement?.parentElement + + if (!height && !width) { + calculateDimensions( + image, + typeof getParentElWidth == 'function' + ? getParentElWidth( + parentElement + ? parseFloat(getComputedStyle(parentElement).width) + : 0, + ) + : getParentElWidth, + ) + } + + image.onload = () => { + setLoad(true) + try { + if (placeholderRef && placeholderRef.current) { + placeholderRef.current.classList.add('hide') + } + + // eslint-disable-next-line no-empty + } catch {} + } + if (showErrorMessage) { + image.onerror = () => { + try { + if (placeholderRef && placeholderRef.current) { + placeholderRef.current.innerHTML = `

图片加载失败!
+ ${escapeHTMLTag(image.src)}

` + } + // eslint-disable-next-line no-empty + } catch {} + } + } + }, [ + src, + loaded, + height, + width, + calculateDimensions, + getParentElWidth, + backgroundColor, + showErrorMessage, + ]) + const memoPlaceholderImage = useMemo( + () => ( + + ), + [backgroundColor, height, width], + ) + + const imageWrapperStyle = useMemo( + () => ({ + height: loaded ? undefined : height || calculatedSize.height, + width: loaded ? undefined : width || calculatedSize.width, + + ...(overflowHidden ? { overflow: 'hidden', borderRadius: '3px' } : {}), + }), + [ + calculatedSize.height, + calculatedSize.width, + height, + loaded, + overflowHidden, + width, + ], + ) + return ( +
+ {defaultImage ? ( + {alt} + ) : ( +
+ + {alt} + {!loaded && memoPlaceholderImage} + +
+ )} + {alt &&
{alt}
} +
+ ) + }), +) + +const PlaceholderImage = memo( + forwardRef< + HTMLDivElement, + { ref: any; className?: string } & Partial + >((props, ref) => { + const { backgroundColor, height, width } = props + return ( +
+ ) + }), +) diff --git a/src/components/ui/image/ZoomedImage.tsx b/src/components/ui/image/ZoomedImage.tsx index f018cffbba..95b6dd99ec 100644 --- a/src/components/ui/image/ZoomedImage.tsx +++ b/src/components/ui/image/ZoomedImage.tsx @@ -1,26 +1,89 @@ -import type { ImageProps } from 'next/image' +'use client' + +import { useCallback, useState } from 'react' +import { tv } from 'tailwind-variants' +import type { ReactNode } from 'react' + +import { LazyLoad } from '~/components/common/Lazyload' +import { useIsUnMounted } from '~/hooks/common/use-is-unmounted' + +import { Divider } from '../divider' type TImageProps = { - className?: string - src?: string - width?: number | string - height?: number | string - 'original-src'?: string - imageRef?: React.MutableRefObject + src: string + alt?: string + title?: string + accent?: string +} + +type BaseImageProps = { zoom?: boolean - accentColor?: string -} & React.HTMLAttributes & - ImageProps + placeholder?: ReactNode +} -export const Image: React.FC = ({ alt, src }) => { - return null +export enum ImageLoadStatus { + Loading = 'loading', + Loaded = 'loaded', + Error = 'error', +} + +const styles = tv({ + base: '', + variants: { + status: { + loading: 'hidden opacity-0', + loaded: 'opacity-100 block', + error: 'hidden opacity-0', + }, + }, +}) +export const ImageLazy: Component = ({ + alt, + src, + title, + accent, + placeholder, +}) => { + const figcaption = title || alt + const [imageLoadStatus, setImageLoadStatus] = useState( + ImageLoadStatus.Loading, + ) + const isUnmount = useIsUnMounted() + const setImageLoadStatusSafe = useCallback( + (status: ImageLoadStatus) => { + if (!isUnmount.current) { + setImageLoadStatus(status) + } + }, + [isUnmount], + ) + return ( + +
+ {alt} setImageLoadStatusSafe(ImageLoadStatus.Loaded)} + onError={() => setImageLoadStatusSafe(ImageLoadStatus.Error)} + className={styles({ + status: imageLoadStatus, + })} + /> +
+ + {figcaption} +
+
+
+ ) } -export const ZoomedImage: React.FC = (props) => { +export const ZoomedImage: Component = (props) => { console.log(props) return ( - + ) } diff --git a/src/components/ui/image/use-calculate-size.tsx b/src/components/ui/image/use-calculate-size.tsx new file mode 100644 index 0000000000..07de70462d --- /dev/null +++ b/src/components/ui/image/use-calculate-size.tsx @@ -0,0 +1,50 @@ +import { useCallback, useReducer } from 'react' + +import { calculateDimensions } from '~/lib/calc-image' + +const initialState = { height: 0, width: 0 } +type Action = { type: 'set'; height: number; width: number } | { type: 'reset' } +export const useCalculateNaturalSize = () => { + const [state, dispatch] = useReducer( + (state: typeof initialState, payload: Action) => { + switch (payload.type) { + case 'set': + return { + height: payload.height, + width: payload.width, + } + case 'reset': + return initialState + default: + return state + } + }, + initialState, + ) + + const calculateOnImageEl = useCallback( + (imageEl: HTMLImageElement, parentElWidth?: number) => { + if (!parentElWidth || !imageEl) { + return + } + + const w = imageEl.naturalWidth, + h = imageEl.naturalHeight + if (w && h) { + const calculated = calculateDimensions(w, h, { + height: Infinity, + width: +parentElWidth, + }) + + dispatch({ + type: 'set', + height: calculated.height, + width: calculated.width, + }) + } + }, + [], + ) + + return [state, calculateOnImageEl] as const +} diff --git a/src/components/ui/markdown/Markdown.tsx b/src/components/ui/markdown/Markdown.tsx index 134b189735..9167eabdfd 100644 --- a/src/components/ui/markdown/Markdown.tsx +++ b/src/components/ui/markdown/Markdown.tsx @@ -1,19 +1,10 @@ /* eslint-disable react-hooks/rules-of-hooks */ -import React, { - createElement, - memo, - useEffect, - useMemo, - useRef, - useState, -} from 'react' +import React, { memo, useMemo, useRef } from 'react' import { clsx } from 'clsx' import { compiler } from 'markdown-to-jsx' import type { MarkdownToJSX } from 'markdown-to-jsx' import type { FC, PropsWithChildren } from 'react' -import { range } from '~/lib/_' - import styles from './index.module.css' import { CommentAtRule } from './parsers/comment-at' import { ContainerRule } from './parsers/container' @@ -25,6 +16,7 @@ import { SpoilderRule } from './parsers/spoiler' import { MParagraph, MTableBody, MTableHead, MTableRow } from './renderers' import { MDetails } from './renderers/collapse' import { MFootNote } from './renderers/footnotes' +import { ZoomedImage } from '../image' export interface MdProps { value?: string @@ -37,7 +29,7 @@ export interface MdProps { > codeBlockFully?: boolean className?: string - tocSlot?: (props: { headings: HTMLElement[] }) => JSX.Element | null + as?: React.ElementType } export const Markdown: FC = @@ -52,30 +44,12 @@ export const Markdown: FC = overrides, extendsRules, additionalParserRules, + as: As = 'div', ...rest } = props const ref = useRef(null) - const [headings, setHeadings] = useState([]) - - useEffect(() => { - if (!ref.current) { - return - } - - const $headings = ref.current.querySelectorAll( - range(1, 6) - .map((i) => `h${i}`) - .join(','), - ) as NodeListOf - - setHeadings(Array.from($headings)) - - return () => { - setHeadings([]) - } - }, [value, props.children]) const node = useMemo(() => { if (!value && typeof props.children != 'string') return null @@ -92,6 +66,7 @@ export const Markdown: FC = // FIXME: footer tag in raw html will renders not as expected, but footer tag in this markdown lib will wrapper as linkReferer footnotes footer: MFootNote, details: MDetails, + img: ZoomedImage, // for custom react component // LinkCard, @@ -163,20 +138,17 @@ export const Markdown: FC = ]) return ( -
- {className ?
{node}
: node} - - {props.tocSlot ? createElement(props.tocSlot, { headings }) : null} -
+ {node} + ) }) diff --git a/src/lib/calc-image.tsx b/src/lib/calc-image.tsx new file mode 100644 index 0000000000..7f7f0f50ed --- /dev/null +++ b/src/lib/calc-image.tsx @@ -0,0 +1,29 @@ +export const calculateDimensions = ( + width: number, + height: number, + max: { width: number; height: number }, +) => { + const { height: maxHeight, width: maxWidth } = max + const wRatio = maxWidth / width + const hRatio = maxHeight / height + let ratio = 1 + if (maxWidth == Infinity && maxHeight == Infinity) { + ratio = 1 + } else if (maxWidth == Infinity) { + if (hRatio < 1) ratio = hRatio + } else if (maxHeight == Infinity) { + if (wRatio < 1) ratio = wRatio + } else if (wRatio < 1 || hRatio < 1) { + ratio = wRatio <= hRatio ? wRatio : hRatio + } + if (ratio < 1) { + return { + width: width * ratio, + height: height * ratio, + } + } + return { + width, + height, + } +} diff --git a/src/lib/fonts.ts b/src/lib/fonts.ts index 1d20aa064c..e00690eaa4 100644 --- a/src/lib/fonts.ts +++ b/src/lib/fonts.ts @@ -1,4 +1,4 @@ -import { Manrope, Noto_Serif } from 'next/font/google' +import { Manrope, Noto_Serif_SC } from 'next/font/google' const sansFont = Manrope({ subsets: ['latin'], @@ -6,7 +6,7 @@ const sansFont = Manrope({ variable: '--font-sans', display: 'swap', }) -const serifFont = Noto_Serif({ +const serifFont = Noto_Serif_SC({ subsets: ['latin'], weight: ['400'], variable: '--font-serif', diff --git a/src/providers/article/article-element-provider.tsx b/src/providers/article/article-element-provider.tsx index 6caf898a7a..a0e5a3f588 100644 --- a/src/providers/article/article-element-provider.tsx +++ b/src/providers/article/article-element-provider.tsx @@ -1,5 +1,6 @@ -import { useEffect, useRef } from 'react' +import { memo, useEffect, useRef } from 'react' import { createContextState } from 'foxact/create-context-state' +import { useIsomorphicLayoutEffect } from 'foxact/use-isomorphic-layout-effect' import { clsxm } from '~/utils/helper' @@ -9,6 +10,15 @@ const [ useSetArticleElement, ] = createContextState(undefined as any) +const [ + ArticleElementSizeProviderInternal, + useArticleElementSize, + useSetArticleElementSize, +] = createContextState({ + h: 0, + w: 0, +}) + const [ IsEOArticleElementProviderInternal, useIsEOArticleElement, @@ -18,23 +28,50 @@ const [ const ArticleElementProvider: Component = ({ children, className }) => { return ( - - {children} - + + + + {children} + + ) } +const ArticleElementResizeObserver = () => { + const setSize = useSetArticleElementSize() + const $article = useArticleElement() + useIsomorphicLayoutEffect(() => { + if (!$article) return + const { height, width } = $article.getBoundingClientRect() + setSize({ h: height, w: width }) + + const observer = new ResizeObserver((entries) => { + const entry = entries[0] + const { height, width } = entry.contentRect + setSize({ h: height, w: width }) + }) + observer.observe($article) + return () => { + observer.unobserve($article) + observer.disconnect() + } + }, [$article]) + + return null +} -const Content: Component = ({ children, className }) => { - const setter = useSetArticleElement() +const Content: Component = memo(({ children, className }) => { + const setElement = useSetArticleElement() return ( -
+
{children}
) -} +}) + +Content.displayName = 'ArticleElementProviderContent' const EOADetector: Component = () => { const ref = useRef(null) @@ -67,4 +104,5 @@ export { useSetArticleElement, useArticleElement, useIsEOArticleElement, + useArticleElementSize, } diff --git a/src/styles/tailwindcss.css b/src/styles/tailwindcss.css index 69be745858..6cf7868a99 100644 --- a/src/styles/tailwindcss.css +++ b/src/styles/tailwindcss.css @@ -29,3 +29,7 @@ html.noise body::before { background-image: url(''); background-repeat: repeat; } + +* { + tab-size: 2; +} diff --git a/src/utils/helper.ts b/src/utils/helper.ts index 4c6b33b1b0..a158d8be4d 100644 --- a/src/utils/helper.ts +++ b/src/utils/helper.ts @@ -4,3 +4,16 @@ import { twMerge } from 'tailwind-merge' export const clsxm = (...args: any[]) => { return twMerge(clsx(args)) } + +export const escapeHTMLTag = (html: string) => { + const lt = //g, + ap = /'/g, + ic = /"/g + return html + .toString() + .replace(lt, '<') + .replace(gt, '>') + .replace(ap, ''') + .replace(ic, '"') +}