('.preview-code-btn');
+
+ for (const button of copyButtons) {
+ button.removeEventListener('click', handleCopyClick);
}
- const tempDiv = document.createElement('div');
- tempDiv.innerHTML = html;
-
- // Make all links open in new tabs
- const linkElements = tempDiv.querySelectorAll('a[href]');
- let mutated = false;
-
- for (const link of linkElements) {
- const target = link.getAttribute('target');
- const rel = link.getAttribute('rel');
-
- if (target !== '_blank' || rel !== 'noopener noreferrer') {
- mutated = true;
- }
-
- link.setAttribute('target', '_blank');
- link.setAttribute('rel', 'noopener noreferrer');
- }
-
- return mutated ? tempDiv.innerHTML : html;
- }
-
- function enhanceCodeBlocks(html: string): string {
- if (!html.includes('
- `;
-
- const actions = document.createElement('div');
- actions.className = 'code-block-actions';
-
- actions.appendChild(copyButton);
-
- if (language.toLowerCase() === 'html') {
- const previewButton = document.createElement('button');
- previewButton.className = 'preview-code-btn';
- previewButton.setAttribute('data-code-id', codeId);
- previewButton.setAttribute('title', 'Preview code');
- previewButton.setAttribute('type', 'button');
-
- previewButton.innerHTML = `
-
- `;
-
- actions.appendChild(previewButton);
- }
-
- header.appendChild(languageLabel);
- header.appendChild(actions);
- wrapper.appendChild(header);
-
- const clonedPre = pre.cloneNode(true) as HTMLElement;
- wrapper.appendChild(clonedPre);
-
- pre.parentNode?.replaceChild(wrapper, pre);
- }
-
- return mutated ? tempDiv.innerHTML : html;
- }
-
- async function processMarkdown(text: string): Promise {
- try {
- let normalized = preprocessLaTeX(text);
- const result = await processor().process(normalized);
- const html = String(result);
- const enhancedLinks = enhanceLinks(html);
-
- return enhanceCodeBlocks(enhancedLinks);
- } catch (error) {
- console.error('Markdown processing error:', error);
-
- // Fallback to plain text with line breaks
- return text.replace(/\n/g, '
');
+ for (const button of previewButtons) {
+ button.removeEventListener('click', handlePreviewClick);
}
}
+ /**
+ * Removes this component's highlight.js theme style from the document head.
+ * Called on component destroy to clean up injected styles.
+ */
+ function cleanupHighlightTheme() {
+ if (!browser) return;
+
+ const existingTheme = document.getElementById(themeStyleId);
+ existingTheme?.remove();
+ }
+
+ /**
+ * Loads the appropriate highlight.js theme based on dark/light mode.
+ * Injects a scoped style element into the document head.
+ * @param isDark - Whether to load the dark theme (true) or light theme (false)
+ */
+ function loadHighlightTheme(isDark: boolean) {
+ if (!browser) return;
+
+ const existingTheme = document.getElementById(themeStyleId);
+ existingTheme?.remove();
+
+ const style = document.createElement('style');
+ style.id = themeStyleId;
+ style.textContent = isDark ? githubDarkCss : githubLightCss;
+
+ document.head.appendChild(style);
+ }
+
+ /**
+ * Extracts code information from a button click target within a code block.
+ * @param target - The clicked button element
+ * @returns Object with rawCode and language, or null if extraction fails
+ */
function getCodeInfoFromTarget(target: HTMLElement) {
const wrapper = target.closest('.code-block-wrapper');
@@ -209,12 +129,7 @@
return null;
}
- const rawCode = codeElement.getAttribute('data-raw-code');
-
- if (rawCode === null) {
- console.error('No raw code found');
- return null;
- }
+ const rawCode = codeElement.textContent ?? '';
const languageLabel = wrapper.querySelector('.code-language');
const language = languageLabel?.textContent?.trim() || 'text';
@@ -222,6 +137,28 @@
return { rawCode, language };
}
+ /**
+ * Generates a unique identifier for a HAST node based on its position.
+ * Used for stable block identification during incremental rendering.
+ * @param node - The HAST root content node
+ * @param indexFallback - Fallback index if position is unavailable
+ * @returns Unique string identifier for the node
+ */
+ function getHastNodeId(node: HastRootContent, indexFallback: number): string {
+ const position = node.position;
+
+ if (position?.start?.offset != null && position?.end?.offset != null) {
+ return `hast-${position.start.offset}-${position.end.offset}`;
+ }
+
+ return `${node.type}-${indexFallback}`;
+ }
+
+ /**
+ * Handles click events on copy buttons within code blocks.
+ * Copies the raw code content to the clipboard.
+ * @param event - The click event from the copy button
+ */
async function handleCopyClick(event: Event) {
event.preventDefault();
event.stopPropagation();
@@ -245,6 +182,25 @@
}
}
+ /**
+ * Handles preview dialog open state changes.
+ * Clears preview content when dialog is closed.
+ * @param open - Whether the dialog is being opened or closed
+ */
+ function handlePreviewDialogOpenChange(open: boolean) {
+ previewDialogOpen = open;
+
+ if (!open) {
+ previewCode = '';
+ previewLanguage = 'text';
+ }
+ }
+
+ /**
+ * Handles click events on preview buttons within HTML code blocks.
+ * Opens a preview dialog with the rendered HTML content.
+ * @param event - The click event from the preview button
+ */
function handlePreviewClick(event: Event) {
event.preventDefault();
event.stopPropagation();
@@ -266,6 +222,61 @@
previewDialogOpen = true;
}
+ /**
+ * Processes markdown content into stable and unstable HTML blocks.
+ * Uses incremental rendering: stable blocks are cached, unstable block is re-rendered.
+ * @param markdown - The raw markdown string to process
+ */
+ async function processMarkdown(markdown: string) {
+ if (!markdown) {
+ renderedBlocks = [];
+ unstableBlockHtml = '';
+ return;
+ }
+
+ const normalized = preprocessLaTeX(markdown);
+ const processorInstance = processor();
+ const ast = processorInstance.parse(normalized) as MdastRoot;
+ const processedRoot = (await processorInstance.run(ast)) as HastRoot;
+ const processedChildren = processedRoot.children ?? [];
+ const stableCount = Math.max(processedChildren.length - 1, 0);
+ const nextBlocks: MarkdownBlock[] = [];
+
+ for (let index = 0; index < stableCount; index++) {
+ const hastChild = processedChildren[index];
+ const id = getHastNodeId(hastChild, index);
+ const existing = renderedBlocks[index];
+
+ if (existing && existing.id === id) {
+ nextBlocks.push(existing);
+ continue;
+ }
+
+ const html = stringifyProcessedNode(
+ processorInstance,
+ processedRoot,
+ processedChildren[index]
+ );
+
+ nextBlocks.push({ id, html });
+ }
+
+ let unstableHtml = '';
+
+ if (processedChildren.length > stableCount) {
+ const unstableChild = processedChildren[stableCount];
+ unstableHtml = stringifyProcessedNode(processorInstance, processedRoot, unstableChild);
+ }
+
+ renderedBlocks = nextBlocks;
+ await tick(); // Force DOM sync before updating unstable HTML block
+ unstableBlockHtml = unstableHtml;
+ }
+
+ /**
+ * Attaches click event listeners to copy and preview buttons in code blocks.
+ * Uses data-listener-bound attribute to prevent duplicate bindings.
+ */
function setupCodeBlockActions() {
if (!containerRef) return;
@@ -287,40 +298,97 @@
}
}
- function handlePreviewDialogOpenChange(open: boolean) {
- previewDialogOpen = open;
+ /**
+ * Converts a single HAST node to an enhanced HTML string.
+ * Applies link and code block enhancements to the output.
+ * @param processorInstance - The remark/rehype processor instance
+ * @param processedRoot - The full processed HAST root (for context)
+ * @param child - The specific HAST child node to stringify
+ * @returns Enhanced HTML string representation of the node
+ */
+ function stringifyProcessedNode(
+ processorInstance: ReturnType,
+ processedRoot: HastRoot,
+ child: unknown
+ ) {
+ const root: HastRoot = {
+ ...(processedRoot as HastRoot),
+ children: [child as never]
+ };
- if (!open) {
- previewCode = '';
- previewLanguage = 'text';
+ return processorInstance.stringify(root);
+ }
+
+ /**
+ * Queues markdown for processing with coalescing support.
+ * Only processes the latest markdown when multiple updates arrive quickly.
+ * @param markdown - The markdown content to render
+ */
+ async function updateRenderedBlocks(markdown: string) {
+ pendingMarkdown = markdown;
+
+ if (isProcessing) {
+ return;
+ }
+
+ isProcessing = true;
+
+ try {
+ while (pendingMarkdown !== null) {
+ const nextMarkdown = pendingMarkdown;
+ pendingMarkdown = null;
+
+ await processMarkdown(nextMarkdown);
+ }
+ } catch (error) {
+ console.error('Failed to process markdown:', error);
+ renderedBlocks = [];
+ unstableBlockHtml = markdown.replace(/\n/g, '
');
+ } finally {
+ isProcessing = false;
}
}
$effect(() => {
- if (content) {
- processMarkdown(content)
- .then((result) => {
- processedHtml = result;
- })
- .catch((error) => {
- console.error('Failed to process markdown:', error);
- processedHtml = content.replace(/\n/g, '
');
- });
- } else {
- processedHtml = '';
- }
+ const currentMode = mode.current;
+ const isDark = currentMode === 'dark';
+
+ loadHighlightTheme(isDark);
});
$effect(() => {
- if (containerRef && processedHtml) {
+ updateRenderedBlocks(content);
+ });
+
+ $effect(() => {
+ const hasRenderedBlocks = renderedBlocks.length > 0;
+ const hasUnstableBlock = Boolean(unstableBlockHtml);
+
+ if ((hasRenderedBlocks || hasUnstableBlock) && containerRef) {
setupCodeBlockActions();
}
});
+
+ onDestroy(() => {
+ cleanupEventListeners();
+ cleanupHighlightTheme();
+ });
-
- {@html processedHtml}
+ {#each renderedBlocks as block (block.id)}
+
+
+ {@html block.html}
+
+ {/each}
+
+ {#if unstableBlockHtml}
+
+
+ {@html unstableBlockHtml}
+
+ {/if}