rowboat/apps/rowboat/app/lib/components/markdown-content.tsx
2025-06-24 15:45:04 +05:30

112 lines
No EOL
5.2 KiB
TypeScript

import Markdown from 'react-markdown'
import remarkGfm from 'remark-gfm'
import { Match } from './mentions_editor';
export default function MarkdownContent({
content,
atValues = []
}: {
content: string;
atValues?: Match[];
}) {
return <div className="overflow-auto break-words">
<Markdown
remarkPlugins={[remarkGfm]}
components={{
h1({ children }) {
return <h1 className="text-xl font-bold py-2">{children}</h1>
},
h2({ children }) {
return <h2 className="text-lg font-bold py-2">{children}</h2>
},
h3({ children }) {
return <h3 className="text-base font-semibold py-2">{children}</h3>
},
h4({ children }) {
return <h4 className="text-sm font-semibold py-2">{children}</h4>
},
h5({ children }) {
return <h5 className="text-xs font-semibold py-2">{children}</h5>
},
h6({ children }) {
return <h6 className="text-xs font-semibold py-2">{children}</h6>
},
strong({ children }) {
return <span className="font-semibold">{children}</span>
},
p({ children }) {
return <p className="py-2">{children}</p>
},
ul({ children }) {
return <ul className="py-2 pl-5 list-disc">{children}</ul>
},
ol({ children }) {
return <ul className="py-2 pl-5 list-decimal">{children}</ul>
},
table({ children }) {
return <table className="py-2 border-collapse border border-gray-400 rounded">{children}</table>
},
th({ children }) {
return <th className="px-2 py-1 border-collapse border border-gray-300 rounded">{children}</th>
},
td({ children }) {
return <td className="px-2 py-1 border-collapse border border-gray-300 rounded">{children}</td>
},
blockquote({ children }) {
return <blockquote className='py-2 bg-gray-200 px-1'>{children}</blockquote>;
},
a(props) {
const { children, href, className, node, ...rest } = props;
// If this is a mention link, render it with mention styling
if (href === '#mention') {
let label: string = '';
// Check if children is an array and get the first text element
if (Array.isArray(children) && children.length > 0) {
const text = children[0];
if (typeof text === 'string') {
const parts = text.split('@');
if (parts.length === 2) {
label = parts[1];
}
}
} else if (typeof children === 'string') {
// Fallback for direct string children
const parts = children.split('@');
if (parts.length === 2) {
label = parts[1];
}
}
// check if the the mention is valid
const invalid = !atValues.some(atValue => atValue.id === label);
if (atValues.length > 0 && invalid) {
return (
<span className="inline-block bg-[#e0f2fe] text-[red] px-1.5 py-0.5 rounded whitespace-nowrap">
@{label} (!)
</span>
);
}
return (
<span className="inline-block bg-[#e0f2fe] text-[#1e40af] px-1.5 py-0.5 rounded whitespace-nowrap">
@{label}
</span>
);
}
// Otherwise render normal link (your existing link component)
return <a className="inline-flex items-center gap-1" target="_blank" href={href} {...rest} >
<span className='underline'>
{children}
</span>
<svg className="w-[16px] h-[16px]" aria-hidden="true" xmlns="http://www.w3.org/2000/svg" width="24" height="24" fill="none" viewBox="0 0 24 24">
<path stroke="currentColor" strokeLinecap="round" strokeLinejoin="round" strokeWidth="1" d="M18 14v4.833A1.166 1.166 0 0 1 16.833 20H5.167A1.167 1.167 0 0 1 4 18.833V7.167A1.166 1.166 0 0 1 5.167 6h4.618m4.447-2H20v5.768m-7.889 2.121 7.778-7.778" />
</svg>
</a>
},
}}
>
{content}
</Markdown>
</div>;
}