mirror of
https://github.com/aljazceru/opencode.git
synced 2026-02-07 00:54:31 +01:00
Co-authored-by: GitHub Action <action@github.com> Co-authored-by: Liang-Shih Lin <liangshihlin@proton.me> Co-authored-by: Dominik Engelhardt <dominikengelhardt@ymail.com> Co-authored-by: Jay V <air@live.ca> Co-authored-by: adamdottv <2363879+adamdottv@users.noreply.github.com>
66 lines
1.7 KiB
TypeScript
66 lines
1.7 KiB
TypeScript
import style from "./content-markdown.module.css"
|
|
import { createResource, createSignal } from "solid-js"
|
|
import { createOverflow } from "./common"
|
|
import { transformerNotationDiff } from "@shikijs/transformers"
|
|
import { marked } from "marked"
|
|
import markedShiki from "marked-shiki"
|
|
import { codeToHtml } from "shiki"
|
|
|
|
const markedWithShiki = marked.use(
|
|
markedShiki({
|
|
highlight(code, lang) {
|
|
return codeToHtml(code, {
|
|
lang: lang || "text",
|
|
themes: {
|
|
light: "github-light",
|
|
dark: "github-dark",
|
|
},
|
|
transformers: [transformerNotationDiff()],
|
|
})
|
|
},
|
|
}),
|
|
)
|
|
|
|
interface Props {
|
|
text: string
|
|
expand?: boolean
|
|
highlight?: boolean
|
|
}
|
|
export function ContentMarkdown(props: Props) {
|
|
const [html] = createResource(
|
|
() => strip(props.text),
|
|
async (markdown) => {
|
|
return markedWithShiki.parse(markdown)
|
|
},
|
|
)
|
|
const [expanded, setExpanded] = createSignal(false)
|
|
const overflow = createOverflow()
|
|
|
|
return (
|
|
<div
|
|
class={style.root}
|
|
data-highlight={props.highlight === true ? true : undefined}
|
|
data-expanded={expanded() || props.expand === true ? true : undefined}
|
|
>
|
|
<div data-slot="markdown" ref={overflow.ref} innerHTML={html()} />
|
|
|
|
{!props.expand && overflow.status && (
|
|
<button
|
|
type="button"
|
|
data-component="text-button"
|
|
data-slot="expand-button"
|
|
onClick={() => setExpanded((e) => !e)}
|
|
>
|
|
{expanded() ? "Show less" : "Show more"}
|
|
</button>
|
|
)}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function strip(text: string): string {
|
|
const wrappedRe = /^\s*<([A-Za-z]\w*)>\s*([\s\S]*?)\s*<\/\1>\s*$/
|
|
const match = text.match(wrappedRe)
|
|
return match ? match[2] : text
|
|
}
|