Skip to content

Commit

Permalink
Refactored the Markdown Components
Browse files Browse the repository at this point in the history
  • Loading branch information
shawkyebrahim2514 committed Nov 29, 2024
1 parent 3095857 commit 02ed27b
Show file tree
Hide file tree
Showing 9 changed files with 192 additions and 151 deletions.
4 changes: 3 additions & 1 deletion react-frontend/src/components/Button/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,10 @@ type ButtonProps = {
readonly size?: keyof typeof differentSizes;
readonly style?: React.CSSProperties;
readonly pointer?: boolean;
readonly children?: JSX.Element[];
}

export default function Button({ icon, text, onClick, size = "lg", style, pointer }: ButtonProps) {
export default function Button({ icon, text, onClick, size = "lg", style, pointer, children }: ButtonProps) {
const { theme } = useThemeContext();
const buttonStyle = useMemo(() => css({
padding: differentSizes[size].padding,
Expand All @@ -50,6 +51,7 @@ export default function Button({ icon, text, onClick, size = "lg", style, pointe
onKeyDown={handleKeyDown} >
{icon}
{text}
{children}
</div >
)
}
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ type AncherLinkMarkdownProps = {
const AncherLinkMarkdown = ({ node, ...props }: AncherLinkMarkdownProps) => {
const title = props.children as string;
const link = node?.properties?.href as string;
// `[[Google]](https://www.google.com)`
const matchButtonLink = /\[(.*)\]/.exec(title);
if (matchButtonLink) {
return (
Expand All @@ -23,6 +24,7 @@ const AncherLinkMarkdown = ({ node, ...props }: AncherLinkMarkdownProps) => {
/>
)
}
// `[Google](https://www.google.com)`
return (
<Header
title={title}
Expand Down
62 changes: 6 additions & 56 deletions react-frontend/src/components/HTMLMarkdown/BlockquoteMarkdown.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { useMemo } from 'react';
import { useThemeContext } from '../../contexts/ThemeContext';
import type { Element, Node, Text, Parent, RootContent } from 'hast'
import type { Element, RootContent } from 'hast'
import MainSection from '../MainSection';
import { toJsxRuntime } from 'hast-util-to-jsx-runtime'
import { Fragment, jsx, jsxs } from 'react/jsx-runtime'
Expand All @@ -16,65 +16,17 @@ type BlockquoteMarkdownProps = {
node?: Element,
} & React.HTMLAttributes<HTMLQuoteElement>;

const isTextNode = (node: Node): node is Text => {
return node?.type === 'text';
};

const isParentNode = (node: Node): node is Parent => {
return node && 'children' in node;
};

const dfsVisit = (currentNode: Node, callback: (node: Node) => boolean | undefined): boolean => {
let isFinished = false;
if (isParentNode(currentNode)) {
for (const child of currentNode.children) {
isFinished = callback(child) || isFinished;
if (!isFinished) {
isFinished = dfsVisit(child, callback) || isFinished;
}
}
}
return isFinished;
};

const BlockquoteMarkdown = ({ node, ...props }: BlockquoteMarkdownProps) => {
const BlockquoteMarkdown = ({ node, className, ...props }: BlockquoteMarkdownProps) => {
const { theme } = useThemeContext();

let { className, title, content } = useMemo(() => {
let className = props?.className?.split(' ') || [];
let title = '';
let content: Node[] = node?.children || [];
let completed = false;
dfsVisit(node as Node, (currentNode: Node) => {
if (completed) {
content.push(currentNode);
} else if (isTextNode(currentNode)) {
if (currentNode.value) {
const match = /\[!(.+)\](.*)/.exec(currentNode.value);
if (match) {
match[1].trim().split(' ').forEach((element) => {
className.push(element);
});
title = match[2].trim();
content = [];
completed = true;
return true;
}
}
}
return false;
});
return { className, title, content };
}, [node, props.className]);

const contentJSXElementsFromAST = useMemo(() => (
content.map((element) => toJsxRuntime(element as RootContent, {
node?.children.map((element) => toJsxRuntime(element as RootContent, {
Fragment, jsx, jsxs, passNode: true, components: {
...markdownComponents,
br: () => null,
}
}))
), [content]);
), [node?.children]);

const targetElement = useMemo((): TargetElementType => ({
"highlight-background": {
Expand Down Expand Up @@ -104,10 +56,9 @@ const BlockquoteMarkdown = ({ node, ...props }: BlockquoteMarkdownProps) => {
let colorType = useMemo((): BlockquoteColorType => className?.includes("secondary") ? "secondary" : "base", [className]);
let colors = useMemo(() => targetElement[backgroundType][colorType], [backgroundType, colorType, targetElement]);

if (className.includes("popup")) {
if (className?.includes("popup")) {
return (
<MainSection
title={title}>
<MainSection>
{contentJSXElementsFromAST}
</MainSection>
);
Expand All @@ -124,7 +75,6 @@ const BlockquoteMarkdown = ({ node, ...props }: BlockquoteMarkdownProps) => {
backgroundColor: `${colors.containerBackgroundColor}`,
color: `${colors.barColor}`
}}>
<h3>{title}</h3>
<div style={{
position: "absolute",
left: 0,
Expand Down
80 changes: 49 additions & 31 deletions react-frontend/src/components/HTMLMarkdown/HeadingMarkdown.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ type HeadingMarkdownProps = React.HTMLAttributes<HTMLHRElement> & {
headingNumber: "one" | "two" | "three" | "four" | "five" | "six",
}

const HeadingMarkdown = ({ headingNumber, ...props }: HeadingMarkdownProps) => {
const HeadingMarkdown = ({ headingNumber, children, ...props }: HeadingMarkdownProps) => {
const { theme } = useThemeContext();
const HTMLHeadingStyles: Record<HeadingMarkdownProps["headingNumber"], React.CSSProperties> = {
one: {
Expand All @@ -30,36 +30,54 @@ const HeadingMarkdown = ({ headingNumber, ...props }: HeadingMarkdownProps) => {
}
}
const HTMLHeadings: Record<HeadingMarkdownProps["headingNumber"], JSX.Element> = {
one: <h1 {...props}
style={{
...props.style,
...HTMLHeadingStyles.one
}} />,
two: <h2 {...props}
style={{
...props.style,
...HTMLHeadingStyles.two
}} />,
three: <h3 {...props}
style={{
...props.style,
...HTMLHeadingStyles.three
}} />,
four: <h4 {...props}
style={{
...props.style,
...HTMLHeadingStyles.four
}} />,
five: <h5 {...props}
style={{
...props.style,
...HTMLHeadingStyles.five
}} />,
six: <h6 {...props}
style={{
...props.style,
...HTMLHeadingStyles.six
}} />
one:
<h1 {...props}
style={{
...props.style,
...HTMLHeadingStyles.one
}}>
{children}
</h1>,
two:
<h2 {...props}
style={{
...props.style,
...HTMLHeadingStyles.two
}}>
{children}
</h2>,
three:
<h3 {...props}
style={{
...props.style,
...HTMLHeadingStyles.three
}}>
{children}
</h3>,
four:
<h4 {...props}
style={{
...props.style,
...HTMLHeadingStyles.four
}}>
{children}
</h4>,
five:
<h5 {...props}
style={{
...props.style,
...HTMLHeadingStyles.five
}}>
{children}
</h5>,
six:
<h6 {...props}
style={{
...props.style,
...HTMLHeadingStyles.six
}}>
{children}
</h6>,
}

return HTMLHeadings[headingNumber];
Expand Down
2 changes: 1 addition & 1 deletion react-frontend/src/components/HTMLMarkdown/HrMarkdown.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ const HrMarkdown = ({ ...props }: React.HTMLAttributes<HTMLHRElement>) => {
margin: "1rem 0",
backgroundColor: theme.colors.base[200],
height: "1px",
}), []);
}), [theme.colors.base]);
return (
<div {...props} style={{
...props.style,
Expand Down
76 changes: 16 additions & 60 deletions react-frontend/src/components/HTMLMarkdown/SpanMarkdown.tsx
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
import { CSSProperties, useMemo } from 'react';
import { useThemeContext } from '../../contexts/ThemeContext';
import type { Element, Text, Node, RootContent } from 'hast'
import { visit } from 'unist-util-visit';
import type { Element, RootContent } from 'hast'
import { v4 as uuidv4 } from 'uuid';
import { Fragment, jsx, jsxs } from 'react/jsx-runtime'
import { markdownComponents } from '.';
import { toJsxRuntime } from 'hast-util-to-jsx-runtime'
import Button from '../Button';

type SpanElementType = "highlight-area" | "highlight-text";
type SpanColorType = "base" | "secondary";
Expand All @@ -15,69 +15,20 @@ type SpanMarkdownProps = {
node?: Element,
} & React.HTMLAttributes<HTMLSpanElement>;

const isTextNode = (node: Node): node is Text => {
return node?.type === 'text';
};

const SpanMarkdown = ({ node, ...props }: SpanMarkdownProps) => {
const SpanMarkdown = ({ node, className, ...props }: SpanMarkdownProps) => {
const { theme } = useThemeContext();
let className = (() => props?.className?.split(' ') || [])();
let content = (() => {
let currentNodes: Node[] = node?.children || [];
// No children element found, only a text node
if (currentNodes.length === 1 && isTextNode(currentNodes[0])) {
let text = currentNodes[0].value;
visit(node as Element, 'text', (textNode: Text) => {
text = textNode.value;
const matchHighlightAreaWithSecondaryColor = text.match(/^!-(.*?)-!/g);
const matchHighlightAreaWithBaseColor = text.match(/^-(.*?)-/g);
const matchHighlightTextWithSecondaryColor = text.match(/^!(.*?)!/g);
if (matchHighlightAreaWithSecondaryColor) {
className.push("highlight-area");
className.push("secondary");
text = text.substring(2, matchHighlightAreaWithSecondaryColor[0].length - 2);
} else if (matchHighlightAreaWithBaseColor) {
className.push("highlight-area");
text = text.substring(1, matchHighlightAreaWithBaseColor[0].length - 1);
} else if (matchHighlightTextWithSecondaryColor) {
className.push("secondary");
text = text.substring(1, matchHighlightTextWithSecondaryColor[0].length - 1);
}
});
return text;
}
let firstNodeChild = currentNodes[0];
let lastNodeChild = currentNodes[currentNodes.length - 1];
if (isTextNode(firstNodeChild) && isTextNode(lastNodeChild) &&
firstNodeChild.value === lastNodeChild.value.split('').reverse().join('')) {
let text = firstNodeChild.value;
const matchHighlightAreaWithSecondaryColor = /^!-/.exec(text);
const matchHighlightAreaWithBaseColor = /^-/.exec(text);
const matchHighlightTextWithSecondaryColor = /^!/.exec(text);
if (matchHighlightAreaWithSecondaryColor) {
className.push("highlight-area");
className.push("secondary");
} else if (matchHighlightAreaWithBaseColor) {
className.push("highlight-area");
} else if (matchHighlightTextWithSecondaryColor) {
className.push("secondary");
}
// Delete the first and last node
currentNodes.shift();
currentNodes.pop();
}
return currentNodes;
})();
let classes = (() => className?.split(' ') || [])();
let content = node?.children;

const contentJSXElementsFromAST = useMemo(() => (
typeof content === 'string' ? content :
content.map((element) => toJsxRuntime(element as RootContent, {
node?.children.map((element) => toJsxRuntime(element as RootContent, {
Fragment, jsx, jsxs, passNode: true, components: {
...markdownComponents,
br: () => null,
}
}))
), [content]);
), [content, node?.children]);

const textStyle = useMemo((): CSSProperties => {
return {
Expand Down Expand Up @@ -161,15 +112,20 @@ const SpanMarkdown = ({ node, ...props }: SpanMarkdownProps) => {
},
}), [lightEffectStyle, textStyle, theme.colors.base, theme.colors.secondary]);
const spanElement = useMemo((): SpanElementType => {
return className?.includes("highlight-area") ? "highlight-area" : "highlight-text";
}, [className]);
return classes.includes("highlight-area") ? "highlight-area" : "highlight-text";
}, [classes]);
const colorType = useMemo((): SpanColorType => {
return className?.includes("secondary") ? "secondary" : "base";
}, [className]);
return classes.includes("secondary") ? "secondary" : "base";
}, [classes]);
const { style, children } = useMemo((): { style: CSSProperties, children: React.ReactNode } => {
return targetElement[spanElement][colorType](props.style ?? {}, contentJSXElementsFromAST);
}, [colorType, contentJSXElementsFromAST, props.style, spanElement, targetElement]);

// `**[[Button]]**`
if(classes.includes("button")) {
return <Button key={classes.join()} size='sm'>{contentJSXElementsFromAST}</Button>
}

return (
<span {...props} style={style}>
{children}
Expand Down
Loading

0 comments on commit 02ed27b

Please sign in to comment.