quikdown 1.1.1 → 1.2.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +39 -7
- package/dist/quikdown.cjs +23 -10
- package/dist/quikdown.dark.css +1 -1
- package/dist/quikdown.esm.js +23 -10
- package/dist/quikdown.esm.min.js +2 -2
- package/dist/quikdown.esm.min.js.gz +0 -0
- package/dist/quikdown.esm.min.js.map +1 -1
- package/dist/quikdown.light.css +1 -1
- package/dist/quikdown.umd.js +23 -10
- package/dist/quikdown.umd.min.js +2 -2
- package/dist/quikdown.umd.min.js.gz +0 -0
- package/dist/quikdown.umd.min.js.map +1 -1
- package/dist/quikdown_ast.cjs +513 -0
- package/dist/quikdown_ast.d.ts +227 -0
- package/dist/quikdown_ast.esm.js +511 -0
- package/dist/quikdown_ast.esm.min.js +8 -0
- package/dist/quikdown_ast.esm.min.js.gz +0 -0
- package/dist/quikdown_ast.esm.min.js.map +1 -0
- package/dist/quikdown_ast.umd.js +519 -0
- package/dist/quikdown_ast.umd.min.js +8 -0
- package/dist/quikdown_ast.umd.min.js.gz +0 -0
- package/dist/quikdown_ast.umd.min.js.map +1 -0
- package/dist/quikdown_ast_html.cjs +1058 -0
- package/dist/quikdown_ast_html.d.ts +68 -0
- package/dist/quikdown_ast_html.esm.js +1056 -0
- package/dist/quikdown_ast_html.esm.min.js +8 -0
- package/dist/quikdown_ast_html.esm.min.js.gz +0 -0
- package/dist/quikdown_ast_html.esm.min.js.map +1 -0
- package/dist/quikdown_ast_html.umd.js +1064 -0
- package/dist/quikdown_ast_html.umd.min.js +8 -0
- package/dist/quikdown_ast_html.umd.min.js.gz +0 -0
- package/dist/quikdown_ast_html.umd.min.js.map +1 -0
- package/dist/quikdown_bd.cjs +38 -19
- package/dist/quikdown_bd.esm.js +38 -19
- package/dist/quikdown_bd.esm.min.js +2 -2
- package/dist/quikdown_bd.esm.min.js.gz +0 -0
- package/dist/quikdown_bd.esm.min.js.map +1 -1
- package/dist/quikdown_bd.umd.js +38 -19
- package/dist/quikdown_bd.umd.min.js +2 -2
- package/dist/quikdown_bd.umd.min.js.gz +0 -0
- package/dist/quikdown_bd.umd.min.js.map +1 -1
- package/dist/quikdown_edit.cjs +836 -117
- package/dist/quikdown_edit.d.ts +123 -131
- package/dist/quikdown_edit.esm.js +836 -117
- package/dist/quikdown_edit.esm.min.js +3 -3
- package/dist/quikdown_edit.esm.min.js.gz +0 -0
- package/dist/quikdown_edit.esm.min.js.map +1 -1
- package/dist/quikdown_edit.umd.js +836 -117
- package/dist/quikdown_edit.umd.min.js +3 -3
- package/dist/quikdown_edit.umd.min.js.gz +0 -0
- package/dist/quikdown_edit.umd.min.js.map +1 -1
- package/dist/quikdown_json.cjs +556 -0
- package/dist/quikdown_json.d.ts +48 -0
- package/dist/quikdown_json.esm.js +554 -0
- package/dist/quikdown_json.esm.min.js +8 -0
- package/dist/quikdown_json.esm.min.js.gz +0 -0
- package/dist/quikdown_json.esm.min.js.map +1 -0
- package/dist/quikdown_json.umd.js +562 -0
- package/dist/quikdown_json.umd.min.js +8 -0
- package/dist/quikdown_json.umd.min.js.gz +0 -0
- package/dist/quikdown_json.umd.min.js.map +1 -0
- package/dist/quikdown_yaml.cjs +717 -0
- package/dist/quikdown_yaml.d.ts +51 -0
- package/dist/quikdown_yaml.esm.js +715 -0
- package/dist/quikdown_yaml.esm.min.js +8 -0
- package/dist/quikdown_yaml.esm.min.js.gz +0 -0
- package/dist/quikdown_yaml.esm.min.js.map +1 -0
- package/dist/quikdown_yaml.umd.js +723 -0
- package/dist/quikdown_yaml.umd.min.js +8 -0
- package/dist/quikdown_yaml.umd.min.js.gz +0 -0
- package/dist/quikdown_yaml.umd.min.js.map +1 -0
- package/package.json +100 -45
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"quikdown_edit.umd.min.js","sources":["../src/quikdown.js","../src/quikdown_bd.js","../src/quikdown_edit_copy.js","../src/quikdown_edit.js"],"sourcesContent":["/**\n * quikdown - A minimal markdown parser optimized for chat/LLM output\n * Supports tables, code blocks, lists, and common formatting\n * @param {string} markdown - The markdown source text\n * @param {Object} options - Optional configuration object\n * @param {Function} options.fence_plugin - Custom renderer for fenced code blocks\n * (content, fence_string) => html string\n * @param {boolean} options.inline_styles - If true, uses inline styles instead of classes\n * @param {boolean} options.bidirectional - If true, adds data-qd attributes for source tracking\n * @param {boolean} options.lazy_linefeeds - If true, single newlines become <br> tags\n * @returns {string} - The rendered HTML\n */\n\n// Version will be injected at build time \nconst quikdownVersion = '__QUIKDOWN_VERSION__';\n\n// Constants for reuse\nconst CLASS_PREFIX = 'quikdown-';\nconst PLACEHOLDER_CB = '§CB';\nconst PLACEHOLDER_IC = '§IC';\n\n// Escape map at module level\nconst ESC_MAP = {'&':'&','<':'<','>':'>','\"':'"',\"'\":'''};\n\n// Single source of truth for all style definitions - optimized\nconst QUIKDOWN_STYLES = {\n h1: 'font-size:2em;font-weight:600;margin:.67em 0;text-align:left',\n h2: 'font-size:1.5em;font-weight:600;margin:.83em 0',\n h3: 'font-size:1.25em;font-weight:600;margin:1em 0',\n h4: 'font-size:1em;font-weight:600;margin:1.33em 0',\n h5: 'font-size:.875em;font-weight:600;margin:1.67em 0',\n h6: 'font-size:.85em;font-weight:600;margin:2em 0',\n pre: 'background:#f4f4f4;padding:10px;border-radius:4px;overflow-x:auto;margin:1em 0',\n code: 'background:#f0f0f0;padding:2px 4px;border-radius:3px;font-family:monospace',\n blockquote: 'border-left:4px solid #ddd;margin-left:0;padding-left:1em',\n table: 'border-collapse:collapse;width:100%;margin:1em 0',\n th: 'border:1px solid #ddd;padding:8px;background-color:#f2f2f2;font-weight:bold;text-align:left',\n td: 'border:1px solid #ddd;padding:8px;text-align:left',\n hr: 'border:none;border-top:1px solid #ddd;margin:1em 0',\n img: 'max-width:100%;height:auto',\n a: 'color:#06c;text-decoration:underline',\n strong: 'font-weight:bold',\n em: 'font-style:italic',\n del: 'text-decoration:line-through',\n ul: 'margin:.5em 0;padding-left:2em',\n ol: 'margin:.5em 0;padding-left:2em',\n li: 'margin:.25em 0',\n // Task list specific styles\n 'task-item': 'list-style:none',\n 'task-checkbox': 'margin-right:.5em'\n};\n\n// Factory function to create getAttr for a given context\nfunction createGetAttr(inline_styles, styles) {\n return function(tag, additionalStyle = '') {\n if (inline_styles) {\n let style = styles[tag];\n if (!style && !additionalStyle) return '';\n \n // Remove default text-align if we're adding a different alignment\n if (additionalStyle && additionalStyle.includes('text-align') && style && style.includes('text-align')) {\n style = style.replace(/text-align:[^;]+;?/, '').trim();\n if (style && !style.endsWith(';')) style += ';';\n }\n \n /* istanbul ignore next - defensive: additionalStyle without style doesn't occur with current tags */\n const fullStyle = additionalStyle ? (style ? `${style}${additionalStyle}` : additionalStyle) : style;\n return ` style=\"${fullStyle}\"`;\n } else {\n const classAttr = ` class=\"${CLASS_PREFIX}${tag}\"`;\n // Apply inline styles for alignment even when using CSS classes\n if (additionalStyle) {\n return `${classAttr} style=\"${additionalStyle}\"`;\n }\n return classAttr;\n }\n };\n}\n\nfunction quikdown(markdown, options = {}) {\n if (!markdown || typeof markdown !== 'string') {\n return '';\n }\n \n const { fence_plugin, inline_styles = false, bidirectional = false, lazy_linefeeds = false } = options;\n const styles = QUIKDOWN_STYLES; // Use module-level styles\n const getAttr = createGetAttr(inline_styles, styles); // Create getAttr once\n\n // Escape HTML entities to prevent XSS\n function escapeHtml(text) {\n return text.replace(/[&<>\"']/g, m => ESC_MAP[m]);\n }\n \n // Helper to add data-qd attributes for bidirectional support\n const dataQd = bidirectional ? (marker) => ` data-qd=\"${escapeHtml(marker)}\"` : () => '';\n \n // Sanitize URLs to prevent XSS attacks\n function sanitizeUrl(url, allowUnsafe = false) {\n /* istanbul ignore next - defensive programming, regex ensures url is never empty */\n if (!url) return '';\n \n // If unsafe URLs are explicitly allowed, return as-is\n if (allowUnsafe) return url;\n \n const trimmedUrl = url.trim();\n const lowerUrl = trimmedUrl.toLowerCase();\n \n // Block dangerous protocols\n const dangerousProtocols = ['javascript:', 'vbscript:', 'data:'];\n \n for (const protocol of dangerousProtocols) {\n if (lowerUrl.startsWith(protocol)) {\n // Exception: Allow data:image/* for images\n if (protocol === 'data:' && lowerUrl.startsWith('data:image/')) {\n return trimmedUrl;\n }\n // Return safe empty link for dangerous protocols\n return '#';\n }\n }\n \n return trimmedUrl;\n }\n\n // Process the markdown in phases\n let html = markdown;\n \n // Phase 1: Extract and protect code blocks and inline code\n const codeBlocks = [];\n const inlineCodes = [];\n \n // Extract fenced code blocks first (supports both ``` and ~~~)\n // Match paired fences - ``` with ``` and ~~~ with ~~~\n // Fence must be at start of line\n html = html.replace(/^(```|~~~)([^\\n]*)\\n([\\s\\S]*?)^\\1$/gm, (match, fence, lang, code) => {\n const placeholder = `${PLACEHOLDER_CB}${codeBlocks.length}§`;\n \n // Trim the language specification\n const langTrimmed = lang ? lang.trim() : '';\n \n // If custom fence plugin is provided, use it (v1.1.0: object format required)\n if (fence_plugin && fence_plugin.render && typeof fence_plugin.render === 'function') {\n codeBlocks.push({\n lang: langTrimmed,\n code: code.trimEnd(),\n custom: true,\n fence: fence,\n hasReverse: !!fence_plugin.reverse\n });\n } else {\n codeBlocks.push({\n lang: langTrimmed,\n code: escapeHtml(code.trimEnd()),\n custom: false,\n fence: fence\n });\n }\n return placeholder;\n });\n \n // Extract inline code\n html = html.replace(/`([^`]+)`/g, (match, code) => {\n const placeholder = `${PLACEHOLDER_IC}${inlineCodes.length}§`;\n inlineCodes.push(escapeHtml(code));\n return placeholder;\n });\n \n // Now escape HTML in the rest of the content\n html = escapeHtml(html);\n \n // Phase 2: Process block elements\n \n // Process tables\n html = processTable(html, getAttr);\n \n // Process headings (supports optional trailing #'s)\n html = html.replace(/^(#{1,6})\\s+(.+?)\\s*#*$/gm, (match, hashes, content) => {\n const level = hashes.length;\n return `<h${level}${getAttr('h' + level)}${dataQd(hashes)}>${content}</h${level}>`;\n });\n \n // Process blockquotes (must handle escaped > since we already escaped HTML)\n html = html.replace(/^>\\s+(.+)$/gm, `<blockquote${getAttr('blockquote')}>$1</blockquote>`);\n // Merge consecutive blockquotes\n html = html.replace(/<\\/blockquote>\\n<blockquote>/g, '\\n');\n \n // Process horizontal rules (allow trailing spaces)\n html = html.replace(/^---+\\s*$/gm, `<hr${getAttr('hr')}>`);\n \n // Process lists\n html = processLists(html, getAttr, inline_styles, bidirectional);\n \n // Phase 3: Process inline elements\n \n // Images (must come before links, with URL sanitization)\n html = html.replace(/!\\[([^\\]]*)\\]\\(([^)]+)\\)/g, (match, alt, src) => {\n const sanitizedSrc = sanitizeUrl(src, options.allow_unsafe_urls);\n const altAttr = bidirectional && alt ? ` data-qd-alt=\"${escapeHtml(alt)}\"` : '';\n const srcAttr = bidirectional ? ` data-qd-src=\"${escapeHtml(src)}\"` : '';\n return `<img${getAttr('img')} src=\"${sanitizedSrc}\" alt=\"${alt}\"${altAttr}${srcAttr}${dataQd('!')}>`;\n });\n \n // Links (with URL sanitization)\n html = html.replace(/\\[([^\\]]+)\\]\\(([^)]+)\\)/g, (match, text, href) => {\n // Sanitize URL to prevent XSS\n const sanitizedHref = sanitizeUrl(href, options.allow_unsafe_urls);\n const isExternal = /^https?:\\/\\//i.test(sanitizedHref);\n const rel = isExternal ? ' rel=\"noopener noreferrer\"' : '';\n const textAttr = bidirectional ? ` data-qd-text=\"${escapeHtml(text)}\"` : '';\n return `<a${getAttr('a')} href=\"${sanitizedHref}\"${rel}${textAttr}${dataQd('[')}>${text}</a>`;\n });\n \n // Autolinks - convert bare URLs to clickable links\n html = html.replace(/(^|\\s)(https?:\\/\\/[^\\s<]+)/g, (match, prefix, url) => {\n const sanitizedUrl = sanitizeUrl(url, options.allow_unsafe_urls);\n return `${prefix}<a${getAttr('a')} href=\"${sanitizedUrl}\" rel=\"noopener noreferrer\">${url}</a>`;\n });\n \n // Process inline formatting (bold, italic, strikethrough)\n const inlinePatterns = [\n [/\\*\\*(.+?)\\*\\*/g, 'strong', '**'],\n [/__(.+?)__/g, 'strong', '__'],\n [/(?<!\\*)\\*(?!\\*)(.+?)(?<!\\*)\\*(?!\\*)/g, 'em', '*'],\n [/(?<!_)_(?!_)(.+?)(?<!_)_(?!_)/g, 'em', '_'],\n [/~~(.+?)~~/g, 'del', '~~']\n ];\n \n inlinePatterns.forEach(([pattern, tag, marker]) => {\n html = html.replace(pattern, `<${tag}${getAttr(tag)}${dataQd(marker)}>$1</${tag}>`);\n });\n \n // Line breaks\n if (lazy_linefeeds) {\n // Lazy linefeeds: single newline becomes <br> (except between paragraphs and after/before block elements)\n const blocks = [];\n let bi = 0;\n \n // Protect tables and lists \n html = html.replace(/<(table|[uo]l)[^>]*>[\\s\\S]*?<\\/\\1>/g, m => {\n blocks[bi] = m;\n return `§B${bi++}§`;\n });\n \n // Handle paragraphs and block elements\n html = html.replace(/\\n\\n+/g, '§P§')\n // After block elements\n .replace(/(<\\/(?:h[1-6]|blockquote|pre)>)\\n/g, '$1§N§')\n .replace(/(<(?:h[1-6]|blockquote|pre|hr)[^>]*>)\\n/g, '$1§N§')\n // Before block elements \n .replace(/\\n(<(?:h[1-6]|blockquote|pre|hr)[^>]*>)/g, '§N§$1')\n .replace(/\\n(§B\\d+§)/g, '§N§$1')\n .replace(/(§B\\d+§)\\n/g, '$1§N§')\n // Convert remaining newlines\n .replace(/\\n/g, `<br${getAttr('br')}>`)\n // Restore\n .replace(/§N§/g, '\\n')\n .replace(/§P§/g, '</p><p>');\n \n // Restore protected blocks\n blocks.forEach((b, i) => html = html.replace(`§B${i}§`, b));\n \n html = '<p>' + html + '</p>';\n } else {\n // Standard: two spaces at end of line for line breaks\n html = html.replace(/ $/gm, `<br${getAttr('br')}>`);\n \n // Paragraphs (double newlines)\n // Don't add </p> after block elements (they're not in paragraphs)\n html = html.replace(/\\n\\n+/g, (match, offset) => {\n // Check if we're after a block element closing tag\n const before = html.substring(0, offset);\n if (before.match(/<\\/(h[1-6]|blockquote|ul|ol|table|pre|hr)>$/)) {\n return '<p>'; // Just open a new paragraph\n }\n return '</p><p>'; // Normal paragraph break\n });\n html = '<p>' + html + '</p>';\n }\n \n // Clean up empty paragraphs and unwrap block elements\n const cleanupPatterns = [\n [/<p><\\/p>/g, ''],\n [/<p>(<h[1-6][^>]*>)/g, '$1'],\n [/(<\\/h[1-6]>)<\\/p>/g, '$1'],\n [/<p>(<blockquote[^>]*>)/g, '$1'],\n [/(<\\/blockquote>)<\\/p>/g, '$1'],\n [/<p>(<ul[^>]*>|<ol[^>]*>)/g, '$1'],\n [/(<\\/ul>|<\\/ol>)<\\/p>/g, '$1'],\n [/<p>(<hr[^>]*>)<\\/p>/g, '$1'],\n [/<p>(<table[^>]*>)/g, '$1'],\n [/(<\\/table>)<\\/p>/g, '$1'],\n [/<p>(<pre[^>]*>)/g, '$1'],\n [/(<\\/pre>)<\\/p>/g, '$1'],\n [new RegExp(`<p>(${PLACEHOLDER_CB}\\\\d+§)<\\/p>`, 'g'), '$1']\n ];\n \n cleanupPatterns.forEach(([pattern, replacement]) => {\n html = html.replace(pattern, replacement);\n });\n \n // Fix orphaned closing </p> tags after block elements\n // When a paragraph follows a block element, ensure it has opening <p>\n html = html.replace(/(<\\/(?:h[1-6]|blockquote|ul|ol|table|pre|hr)>)\\n([^<])/g, '$1\\n<p>$2');\n \n // Phase 4: Restore code blocks and inline code\n \n // Restore code blocks\n codeBlocks.forEach((block, i) => {\n let replacement;\n \n if (block.custom && fence_plugin && fence_plugin.render) {\n // Use custom fence plugin (v1.1.0: object format with render function)\n replacement = fence_plugin.render(block.code, block.lang);\n \n // If plugin returns undefined, fall back to default rendering\n if (replacement === undefined) {\n const langClass = !inline_styles && block.lang ? ` class=\"language-${block.lang}\"` : '';\n const codeAttr = inline_styles ? getAttr('code') : langClass;\n const langAttr = bidirectional && block.lang ? ` data-qd-lang=\"${escapeHtml(block.lang)}\"` : '';\n const fenceAttr = bidirectional ? ` data-qd-fence=\"${escapeHtml(block.fence)}\"` : '';\n replacement = `<pre${getAttr('pre')}${fenceAttr}${langAttr}><code${codeAttr}>${escapeHtml(block.code)}</code></pre>`;\n } else if (bidirectional) {\n // If bidirectional and plugin provided HTML, add data attributes for roundtrip\n replacement = replacement.replace(/^<(\\w+)/, \n `<$1 data-qd-fence=\"${escapeHtml(block.fence)}\" data-qd-lang=\"${escapeHtml(block.lang)}\" data-qd-source=\"${escapeHtml(block.code)}\"`);\n }\n } else {\n // Default rendering\n const langClass = !inline_styles && block.lang ? ` class=\"language-${block.lang}\"` : '';\n const codeAttr = inline_styles ? getAttr('code') : langClass;\n const langAttr = bidirectional && block.lang ? ` data-qd-lang=\"${escapeHtml(block.lang)}\"` : '';\n const fenceAttr = bidirectional ? ` data-qd-fence=\"${escapeHtml(block.fence)}\"` : '';\n replacement = `<pre${getAttr('pre')}${fenceAttr}${langAttr}><code${codeAttr}>${block.code}</code></pre>`;\n }\n \n const placeholder = `${PLACEHOLDER_CB}${i}§`;\n html = html.replace(placeholder, replacement);\n });\n \n // Restore inline code\n inlineCodes.forEach((code, i) => {\n const placeholder = `${PLACEHOLDER_IC}${i}§`;\n html = html.replace(placeholder, `<code${getAttr('code')}${dataQd('`')}>${code}</code>`);\n });\n \n return html.trim();\n}\n\n/**\n * Process inline markdown formatting\n */\nfunction processInlineMarkdown(text, getAttr) {\n \n // Process inline formatting patterns\n const patterns = [\n [/\\*\\*(.+?)\\*\\*/g, 'strong'],\n [/__(.+?)__/g, 'strong'],\n [/(?<!\\*)\\*(?!\\*)(.+?)(?<!\\*)\\*(?!\\*)/g, 'em'],\n [/(?<!_)_(?!_)(.+?)(?<!_)_(?!_)/g, 'em'],\n [/~~(.+?)~~/g, 'del'],\n [/`([^`]+)`/g, 'code']\n ];\n \n patterns.forEach(([pattern, tag]) => {\n text = text.replace(pattern, `<${tag}${getAttr(tag)}>$1</${tag}>`);\n });\n \n return text;\n}\n\n/**\n * Process markdown tables\n */\nfunction processTable(text, getAttr) {\n const lines = text.split('\\n');\n const result = [];\n let inTable = false;\n let tableLines = [];\n \n for (let i = 0; i < lines.length; i++) {\n const line = lines[i].trim();\n \n // Check if this line looks like a table row (with or without trailing |)\n if (line.includes('|') && (line.startsWith('|') || /[^\\\\|]/.test(line))) {\n if (!inTable) {\n inTable = true;\n tableLines = [];\n }\n tableLines.push(line);\n } else {\n // Not a table line\n if (inTable) {\n // Process the accumulated table\n const tableHtml = buildTable(tableLines, getAttr);\n if (tableHtml) {\n result.push(tableHtml);\n } else {\n // Not a valid table, restore original lines\n result.push(...tableLines);\n }\n inTable = false;\n tableLines = [];\n }\n result.push(lines[i]);\n }\n }\n \n // Handle table at end of text\n if (inTable && tableLines.length > 0) {\n const tableHtml = buildTable(tableLines, getAttr);\n if (tableHtml) {\n result.push(tableHtml);\n } else {\n result.push(...tableLines);\n }\n }\n \n return result.join('\\n');\n}\n\n/**\n * Build an HTML table from markdown table lines\n */\nfunction buildTable(lines, getAttr) {\n \n if (lines.length < 2) return null;\n \n // Check for separator line (second line should be the separator)\n let separatorIndex = -1;\n for (let i = 1; i < lines.length; i++) {\n // Support separator with or without leading/trailing pipes\n if (/^\\|?[\\s\\-:|]+\\|?$/.test(lines[i]) && lines[i].includes('-')) {\n separatorIndex = i;\n break;\n }\n }\n \n if (separatorIndex === -1) return null;\n \n const headerLines = lines.slice(0, separatorIndex);\n const bodyLines = lines.slice(separatorIndex + 1);\n \n // Parse alignment from separator\n const separator = lines[separatorIndex];\n // Handle pipes at start/end or not\n const separatorCells = separator.trim().replace(/^\\|/, '').replace(/\\|$/, '').split('|');\n const alignments = separatorCells.map(cell => {\n const trimmed = cell.trim();\n if (trimmed.startsWith(':') && trimmed.endsWith(':')) return 'center';\n if (trimmed.endsWith(':')) return 'right';\n return 'left';\n });\n \n let html = `<table${getAttr('table')}>\\n`;\n \n // Build header\n // Note: headerLines will always have length > 0 since separatorIndex starts from 1\n html += `<thead${getAttr('thead')}>\\n`;\n headerLines.forEach(line => {\n html += `<tr${getAttr('tr')}>\\n`;\n // Handle pipes at start/end or not\n const cells = line.trim().replace(/^\\|/, '').replace(/\\|$/, '').split('|');\n cells.forEach((cell, i) => {\n const alignStyle = alignments[i] && alignments[i] !== 'left' ? `text-align:${alignments[i]}` : '';\n const processedCell = processInlineMarkdown(cell.trim(), getAttr);\n html += `<th${getAttr('th', alignStyle)}>${processedCell}</th>\\n`;\n });\n html += '</tr>\\n';\n });\n html += '</thead>\\n';\n \n // Build body\n if (bodyLines.length > 0) {\n html += `<tbody${getAttr('tbody')}>\\n`;\n bodyLines.forEach(line => {\n html += `<tr${getAttr('tr')}>\\n`;\n // Handle pipes at start/end or not\n const cells = line.trim().replace(/^\\|/, '').replace(/\\|$/, '').split('|');\n cells.forEach((cell, i) => {\n const alignStyle = alignments[i] && alignments[i] !== 'left' ? `text-align:${alignments[i]}` : '';\n const processedCell = processInlineMarkdown(cell.trim(), getAttr);\n html += `<td${getAttr('td', alignStyle)}>${processedCell}</td>\\n`;\n });\n html += '</tr>\\n';\n });\n html += '</tbody>\\n';\n }\n \n html += '</table>';\n return html;\n}\n\n/**\n * Process markdown lists (ordered and unordered)\n */\nfunction processLists(text, getAttr, inline_styles, bidirectional) {\n \n const lines = text.split('\\n');\n const result = [];\n let listStack = []; // Track nested lists\n \n // Helper to escape HTML for data-qd attributes\n const escapeHtml = (text) => text.replace(/[&<>\"']/g, m => ({'&':'&','<':'<','>':'>','\"':'"',\"'\":'''})[m]);\n const dataQd = bidirectional ? (marker) => ` data-qd=\"${escapeHtml(marker)}\"` : () => '';\n \n for (let i = 0; i < lines.length; i++) {\n const line = lines[i];\n const match = line.match(/^(\\s*)([*\\-+]|\\d+\\.)\\s+(.+)$/);\n \n if (match) {\n const [, indent, marker, content] = match;\n const level = Math.floor(indent.length / 2);\n const isOrdered = /^\\d+\\./.test(marker);\n const listType = isOrdered ? 'ol' : 'ul';\n \n // Check for task list items\n let listItemContent = content;\n let taskListClass = '';\n const taskMatch = content.match(/^\\[([x ])\\]\\s+(.*)$/i);\n if (taskMatch && !isOrdered) {\n const [, checked, taskContent] = taskMatch;\n const isChecked = checked.toLowerCase() === 'x';\n const checkboxAttr = inline_styles \n ? ' style=\"margin-right:.5em\"' \n : ` class=\"${CLASS_PREFIX}task-checkbox\"`;\n listItemContent = `<input type=\"checkbox\"${checkboxAttr}${isChecked ? ' checked' : ''} disabled> ${taskContent}`;\n taskListClass = inline_styles ? ' style=\"list-style:none\"' : ` class=\"${CLASS_PREFIX}task-item\"`;\n }\n \n // Close deeper levels\n while (listStack.length > level + 1) {\n const list = listStack.pop();\n result.push(`</${list.type}>`);\n }\n \n // Open new level if needed\n if (listStack.length === level) {\n // Need to open a new list\n listStack.push({ type: listType, level });\n result.push(`<${listType}${getAttr(listType)}>`);\n } else if (listStack.length === level + 1) {\n // Check if we need to switch list type\n const currentList = listStack[listStack.length - 1];\n if (currentList.type !== listType) {\n result.push(`</${currentList.type}>`);\n listStack.pop();\n listStack.push({ type: listType, level });\n result.push(`<${listType}${getAttr(listType)}>`);\n }\n }\n \n const liAttr = taskListClass || getAttr('li');\n result.push(`<li${liAttr}${dataQd(marker)}>${listItemContent}</li>`);\n } else {\n // Not a list item, close all lists\n while (listStack.length > 0) {\n const list = listStack.pop();\n result.push(`</${list.type}>`);\n }\n result.push(line);\n }\n }\n \n // Close any remaining lists\n while (listStack.length > 0) {\n const list = listStack.pop();\n result.push(`</${list.type}>`);\n }\n \n return result.join('\\n');\n}\n\n/**\n * Emit CSS styles for quikdown elements\n * @param {string} prefix - Optional class prefix (default: 'quikdown-')\n * @param {string} theme - Optional theme: 'light' (default) or 'dark'\n * @returns {string} CSS string with quikdown styles\n */\nquikdown.emitStyles = function(prefix = 'quikdown-', theme = 'light') {\n const styles = QUIKDOWN_STYLES;\n \n // Define theme color overrides\n const themeOverrides = {\n dark: {\n '#f4f4f4': '#2a2a2a', // pre background\n '#f0f0f0': '#2a2a2a', // code background\n '#f2f2f2': '#2a2a2a', // th background\n '#ddd': '#3a3a3a', // borders\n '#06c': '#6db3f2', // links\n _textColor: '#e0e0e0'\n },\n light: {\n _textColor: '#333' // Explicit text color for light theme\n }\n };\n \n let css = '';\n for (const [tag, style] of Object.entries(styles)) {\n let themedStyle = style;\n \n // Apply theme overrides if dark theme\n if (theme === 'dark' && themeOverrides.dark) {\n // Replace colors\n for (const [oldColor, newColor] of Object.entries(themeOverrides.dark)) {\n if (!oldColor.startsWith('_')) {\n themedStyle = themedStyle.replace(new RegExp(oldColor, 'g'), newColor);\n }\n }\n \n // Add text color for certain elements in dark theme\n const needsTextColor = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'td', 'li', 'blockquote'];\n if (needsTextColor.includes(tag)) {\n themedStyle += `;color:${themeOverrides.dark._textColor}`;\n }\n } else if (theme === 'light' && themeOverrides.light) {\n // Add explicit text color for light theme elements too\n const needsTextColor = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'td', 'li', 'blockquote'];\n if (needsTextColor.includes(tag)) {\n themedStyle += `;color:${themeOverrides.light._textColor}`;\n }\n }\n \n css += `.${prefix}${tag} { ${themedStyle} }\\n`;\n }\n \n return css;\n};\n\n/**\n * Configure quikdown with options and return a function\n * @param {Object} options - Configuration options\n * @returns {Function} Configured quikdown function\n */\nquikdown.configure = function(options) {\n return function(markdown) {\n return quikdown(markdown, options);\n };\n};\n\n/**\n * Version information\n */\nquikdown.version = quikdownVersion;\n\n// Export for both CommonJS and ES6\n/* istanbul ignore next */\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = quikdown;\n}\n\n// For browser global\n/* istanbul ignore next */\nif (typeof window !== 'undefined') {\n window.quikdown = quikdown;\n}\n\nexport default quikdown;","/**\n * quikdown_bd - Bidirectional markdown/HTML converter\n * Extends core quikdown with HTML→Markdown conversion\n * \n * Uses data-qd attributes to preserve original markdown syntax\n * Enables HTML→Markdown conversion for quikdown-generated HTML\n */\n\nimport quikdown from './quikdown.js';\n\n/**\n * Create bidirectional version by extending quikdown\n * This wraps quikdown and adds the toMarkdown method\n */\nfunction quikdown_bd(markdown, options = {}) {\n // Use core quikdown with bidirectional flag to add data-qd attributes\n return quikdown(markdown, { ...options, bidirectional: true });\n}\n\n// Copy all properties and methods from quikdown (including version)\nObject.keys(quikdown).forEach(key => {\n quikdown_bd[key] = quikdown[key];\n});\n\n// Add the toMarkdown method for HTML→Markdown conversion\nquikdown_bd.toMarkdown = function(htmlOrElement, options = {}) {\n // Accept either HTML string or DOM element\n let container;\n if (typeof htmlOrElement === 'string') {\n container = document.createElement('div');\n container.innerHTML = htmlOrElement;\n } else if (htmlOrElement instanceof Element) {\n /* istanbul ignore next - browser-only code path, not testable in jsdom */\n container = htmlOrElement;\n } else {\n return '';\n }\n \n // Walk the DOM tree and reconstruct markdown\n function walkNode(node, parentContext = {}) {\n if (node.nodeType === Node.TEXT_NODE) {\n // Return text content, preserving whitespace where needed\n return node.textContent;\n }\n \n if (node.nodeType !== Node.ELEMENT_NODE) {\n return '';\n }\n \n const tag = node.tagName.toLowerCase();\n const dataQd = node.getAttribute('data-qd');\n \n // Process children with context\n let childContent = '';\n for (let child of node.childNodes) {\n childContent += walkNode(child, { parentTag: tag, ...parentContext });\n }\n \n // Determine markdown based on element and attributes\n switch (tag) {\n case 'h1':\n case 'h2':\n case 'h3':\n case 'h4':\n case 'h5':\n case 'h6':\n const level = parseInt(tag[1]);\n const prefix = dataQd || '#'.repeat(level);\n return `${prefix} ${childContent.trim()}\\n\\n`;\n \n case 'strong':\n case 'b':\n if (!childContent) return ''; // Don't add markers for empty content\n const boldMarker = dataQd || '**';\n return `${boldMarker}${childContent}${boldMarker}`;\n \n case 'em':\n case 'i':\n if (!childContent) return ''; // Don't add markers for empty content\n const emMarker = dataQd || '*';\n return `${emMarker}${childContent}${emMarker}`;\n \n case 'del':\n case 's':\n case 'strike':\n if (!childContent) return ''; // Don't add markers for empty content\n const delMarker = dataQd || '~~';\n return `${delMarker}${childContent}${delMarker}`;\n \n case 'code':\n // Note: code inside pre is handled directly by the pre case using querySelector\n if (!childContent) return ''; // Don't add markers for empty content\n const codeMarker = dataQd || '`';\n return `${codeMarker}${childContent}${codeMarker}`;\n \n case 'pre':\n const fence = node.getAttribute('data-qd-fence') || dataQd || '```';\n const lang = node.getAttribute('data-qd-lang') || '';\n \n // Check if this was created by a fence plugin with reverse handler\n if (options.fence_plugin && options.fence_plugin.reverse && lang) {\n try {\n const result = options.fence_plugin.reverse(node);\n if (result && result.content) {\n const fenceMarker = result.fence || fence;\n const langStr = result.lang || lang;\n return `${fenceMarker}${langStr}\\n${result.content}\\n${fenceMarker}\\n\\n`;\n }\n } catch (err) {\n console.warn('Fence reverse handler error:', err);\n // Fall through to default handling\n }\n }\n \n // Fallback: use data-qd-source if available\n const source = node.getAttribute('data-qd-source');\n if (source) {\n return `${fence}${lang}\\n${source}\\n${fence}\\n\\n`;\n }\n \n // Final fallback: extract text content\n const codeEl = node.querySelector('code');\n const codeContent = codeEl ? codeEl.textContent : childContent;\n return `${fence}${lang}\\n${codeContent.trimEnd()}\\n${fence}\\n\\n`;\n \n case 'blockquote':\n const quoteMarker = dataQd || '>';\n const lines = childContent.trim().split('\\n');\n return lines.map(line => `${quoteMarker} ${line}`).join('\\n') + '\\n\\n';\n \n case 'hr':\n const hrMarker = dataQd || '---';\n return `${hrMarker}\\n\\n`;\n \n case 'br':\n const brMarker = dataQd || ' ';\n return `${brMarker}\\n`;\n \n case 'a':\n const linkText = node.getAttribute('data-qd-text') || childContent.trim();\n const href = node.getAttribute('href') || '';\n // Check for autolinks\n if (linkText === href && !dataQd) {\n return `<${href}>`;\n }\n return `[${linkText}](${href})`;\n \n case 'img':\n const alt = node.getAttribute('data-qd-alt') || node.getAttribute('alt') || '';\n const src = node.getAttribute('data-qd-src') || node.getAttribute('src') || '';\n const imgMarker = dataQd || '!';\n return `${imgMarker}[${alt}](${src})`;\n \n case 'ul':\n case 'ol':\n return walkList(node, tag === 'ol') + '\\n';\n \n case 'li':\n // Handled by list processor\n return childContent;\n \n case 'table':\n return walkTable(node) + '\\n\\n';\n \n case 'p':\n // Check if it's actually a paragraph or just a wrapper\n if (childContent.trim()) {\n // Check if paragraph ends with a line that's just whitespace\n // This indicates an intentional blank line before the next element\n const lines = childContent.split('\\n');\n let content = childContent.trim();\n \n // If the last line(s) are just whitespace, preserve one blank line\n if (lines.length > 1) {\n let trailingBlankLines = 0;\n for (let i = lines.length - 1; i >= 0; i--) {\n if (lines[i].trim() === '') {\n trailingBlankLines++;\n } else {\n break;\n }\n }\n if (trailingBlankLines > 0) {\n // Add a line with just a space, followed by single newline\n // The \\n\\n will be added below for paragraph separation\n content = content + '\\n ';\n // Only add one newline since we're preserving the space line\n return content + '\\n';\n }\n }\n \n return content + '\\n\\n';\n }\n return '';\n \n case 'div':\n // Check if this was created by a fence plugin with reverse handler\n const divLang = node.getAttribute('data-qd-lang');\n const divFence = node.getAttribute('data-qd-fence');\n \n if (divLang && options.fence_plugin && options.fence_plugin.reverse) {\n try {\n const result = options.fence_plugin.reverse(node);\n if (result && result.content) {\n const fenceMarker = result.fence || divFence || '```';\n const langStr = result.lang || divLang;\n return `${fenceMarker}${langStr}\\n${result.content}\\n${fenceMarker}\\n\\n`;\n }\n } catch (err) {\n console.warn('Fence reverse handler error:', err);\n // Fall through to default handling\n }\n }\n \n // Fallback: use data-qd-source if available\n const divSource = node.getAttribute('data-qd-source');\n if (divSource && divFence) {\n return `${divFence}${divLang || ''}\\n${divSource}\\n${divFence}\\n\\n`;\n }\n \n // Check if it's a mermaid container\n if (node.classList && node.classList.contains('mermaid-container')) {\n const fence = node.getAttribute('data-qd-fence') || '```';\n const lang = node.getAttribute('data-qd-lang') || 'mermaid';\n \n // First check for data-qd-source attribute on the container\n const source = node.getAttribute('data-qd-source');\n if (source) {\n // Decode HTML entities from the attribute (mainly ")\n const temp = document.createElement('textarea');\n temp.innerHTML = source;\n const code = temp.value;\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n \n // Check for source on the pre.mermaid element\n const mermaidPre = node.querySelector('pre.mermaid');\n if (mermaidPre) {\n const preSource = mermaidPre.getAttribute('data-qd-source');\n if (preSource) {\n const temp = document.createElement('textarea');\n temp.innerHTML = preSource;\n const code = temp.value;\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n }\n \n // Fallback: Look for the legacy .mermaid-source element\n const sourceElement = node.querySelector('.mermaid-source');\n if (sourceElement) {\n // Decode HTML entities\n const temp = document.createElement('div');\n temp.innerHTML = sourceElement.innerHTML;\n const code = temp.textContent;\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n \n // Final fallback: try to extract from the mermaid element (unreliable after rendering)\n const mermaidElement = node.querySelector('.mermaid');\n if (mermaidElement && mermaidElement.textContent.includes('graph')) {\n return `${fence}${lang}\\n${mermaidElement.textContent.trim()}\\n${fence}\\n\\n`;\n }\n }\n // Check if it's a standalone mermaid diagram (legacy)\n if (node.classList && node.classList.contains('mermaid')) {\n const fence = node.getAttribute('data-qd-fence') || '```';\n const lang = node.getAttribute('data-qd-lang') || 'mermaid';\n const code = node.textContent.trim();\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n // Pass through other divs\n return childContent;\n \n case 'span':\n // Pass through container elements\n return childContent;\n \n default:\n return childContent;\n }\n }\n \n // Walk list elements\n function walkList(listNode, isOrdered, depth = 0) {\n let result = '';\n let index = 1;\n const indent = ' '.repeat(depth);\n \n for (let child of listNode.children) {\n if (child.tagName !== 'LI') continue;\n \n const dataQd = child.getAttribute('data-qd');\n let marker = dataQd || (isOrdered ? `${index}.` : '-');\n \n // Check for task list checkbox\n const checkbox = child.querySelector('input[type=\"checkbox\"]');\n if (checkbox) {\n const checked = checkbox.checked ? 'x' : ' ';\n marker = '-';\n // Get text without the checkbox\n let text = '';\n for (let node of child.childNodes) {\n if (node.nodeType === Node.TEXT_NODE) {\n text += node.textContent;\n } else if (node.tagName && node.tagName !== 'INPUT') {\n text += walkNode(node);\n }\n }\n result += `${indent}${marker} [${checked}] ${text.trim()}\\n`;\n } else {\n let itemContent = '';\n \n for (let node of child.childNodes) {\n if (node.tagName === 'UL' || node.tagName === 'OL') {\n itemContent += walkList(node, node.tagName === 'OL', depth + 1);\n } else {\n itemContent += walkNode(node);\n }\n }\n \n result += `${indent}${marker} ${itemContent.trim()}\\n`;\n }\n \n index++;\n }\n \n return result;\n }\n \n // Walk table elements\n function walkTable(table) {\n let result = '';\n const alignData = table.getAttribute('data-qd-align');\n const alignments = alignData ? alignData.split(',') : [];\n \n // Process header\n const thead = table.querySelector('thead');\n if (thead) {\n const headerRow = thead.querySelector('tr');\n if (headerRow) {\n const headers = [];\n for (let th of headerRow.querySelectorAll('th')) {\n headers.push(th.textContent.trim());\n }\n result += '| ' + headers.join(' | ') + ' |\\n';\n \n // Add separator with alignment\n const separators = headers.map((_, i) => {\n const align = alignments[i] || 'left';\n if (align === 'center') return ':---:';\n if (align === 'right') return '---:';\n return '---';\n });\n result += '| ' + separators.join(' | ') + ' |\\n';\n }\n }\n \n // Process body\n const tbody = table.querySelector('tbody');\n if (tbody) {\n for (let row of tbody.querySelectorAll('tr')) {\n const cells = [];\n for (let td of row.querySelectorAll('td')) {\n cells.push(td.textContent.trim());\n }\n if (cells.length > 0) {\n result += '| ' + cells.join(' | ') + ' |\\n';\n }\n }\n }\n \n return result.trim();\n }\n \n // Process the DOM tree\n let markdown = walkNode(container);\n \n // Clean up\n markdown = markdown.replace(/\\n{3,}/g, '\\n\\n'); // Remove excessive newlines\n markdown = markdown.trim();\n \n return markdown;\n};\n\n// Override the configure method to return a bidirectional version\nquikdown_bd.configure = function(options) {\n return function(markdown) {\n return quikdown_bd(markdown, options);\n };\n};\n\n// Set version\n// Version is already copied from quikdown via Object.keys loop\n\n// Export for both module and browser\n/* istanbul ignore next */\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = quikdown_bd;\n}\n\n/* istanbul ignore next */\nif (typeof window !== 'undefined') {\n window.quikdown_bd = quikdown_bd;\n}\n\nexport default quikdown_bd;","/**\n * Rich copy functionality for QuikdownEditor\n * Handles copying rendered content with proper formatting for pasting into rich text editors\n */\n\n/**\n * Get platform information\n * @returns {string} The detected platform: 'macos', 'windows', 'linux', or 'unknown'\n */\nfunction getPlatform() {\n const platform = navigator.platform?.toLowerCase() || '';\n const userAgent = navigator.userAgent?.toLowerCase() || '';\n \n if (platform.includes('mac') || userAgent.includes('mac')) {\n return 'macos';\n } else if (userAgent.includes('windows')) {\n return 'windows';\n } else if (userAgent.includes('linux')) {\n return 'linux';\n }\n return 'unknown';\n}\n\n/**\n * Copy to clipboard using HTML selection fallback (for Safari)\n * Uses div with selection to preserve HTML formatting\n * @param {string} html - HTML content to copy\n * @returns {boolean} Success status\n */\nfunction copyToClipboard(html) {\n let tempDiv;\n let result;\n \n try {\n // Use a div instead of textarea to preserve HTML formatting\n tempDiv = document.createElement('div');\n tempDiv.style.position = 'fixed';\n tempDiv.style.left = '-9999px';\n tempDiv.style.top = '0';\n tempDiv.style.width = '1px';\n tempDiv.style.height = '1px';\n tempDiv.style.overflow = 'hidden';\n tempDiv.innerHTML = html;\n \n document.body.appendChild(tempDiv);\n \n // Select the HTML content\n const range = document.createRange();\n range.selectNodeContents(tempDiv);\n const selection = window.getSelection();\n selection.removeAllRanges();\n selection.addRange(range);\n \n // Try to copy\n result = document.execCommand('copy');\n \n // Clear selection\n selection.removeAllRanges();\n } catch (err) {\n console.error('Fallback copy failed:', err);\n result = false;\n } finally {\n if (tempDiv && tempDiv.parentNode) {\n document.body.removeChild(tempDiv);\n }\n }\n \n return result;\n}\n\n/**\n * Convert SVG to PNG blob (based on squibview's implementation)\n * @param {SVGElement} svgElement - The SVG element to convert\n * @returns {Promise<Blob>} A promise that resolves with the PNG blob\n */\nasync function svgToPng(svgElement, needsWhiteBackground = false) {\n return new Promise((resolve, reject) => {\n const svgString = new XMLSerializer().serializeToString(svgElement);\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n const img = new Image();\n \n const scale = 2;\n \n // Check if this is a Mermaid-generated SVG (they don't have explicit width/height attributes)\n const isMermaidSvg = svgElement.closest('.mermaid') || svgElement.classList.contains('mermaid');\n const hasExplicitDimensions = svgElement.getAttribute('width') && svgElement.getAttribute('height');\n \n let svgWidth, svgHeight;\n \n if (isMermaidSvg || !hasExplicitDimensions) {\n // For Mermaid or other generated SVGs, prioritize computed dimensions\n svgWidth = svgElement.clientWidth || \n (svgElement.viewBox && svgElement.viewBox.baseVal.width) || \n parseFloat(svgElement.getAttribute('width')) || 400;\n svgHeight = svgElement.clientHeight || \n (svgElement.viewBox && svgElement.viewBox.baseVal.height) || \n parseFloat(svgElement.getAttribute('height')) || 300;\n } else {\n // For explicit SVGs (like fenced SVG blocks), prioritize explicit attributes\n svgWidth = parseFloat(svgElement.getAttribute('width')) || \n (svgElement.viewBox && svgElement.viewBox.baseVal.width) || \n svgElement.clientWidth || 400;\n svgHeight = parseFloat(svgElement.getAttribute('height')) || \n (svgElement.viewBox && svgElement.viewBox.baseVal.height) || \n svgElement.clientHeight || 300;\n }\n \n // Ensure the SVG string has explicit dimensions by modifying it if necessary\n let modifiedSvgString = svgString;\n if (svgWidth && svgHeight) {\n // Create a temporary SVG element to modify the serialized string\n const tempDiv = document.createElement('div');\n tempDiv.innerHTML = svgString;\n const tempSvg = tempDiv.querySelector('svg');\n if (tempSvg) {\n tempSvg.setAttribute('width', svgWidth.toString());\n tempSvg.setAttribute('height', svgHeight.toString());\n modifiedSvgString = new XMLSerializer().serializeToString(tempSvg);\n }\n }\n \n canvas.width = svgWidth * scale;\n canvas.height = svgHeight * scale;\n ctx.scale(scale, scale);\n \n img.onload = () => {\n try {\n // Add white background for math equations (they often have transparent backgrounds)\n if (needsWhiteBackground) {\n ctx.fillStyle = 'white';\n ctx.fillRect(0, 0, canvas.width, canvas.height);\n }\n \n ctx.drawImage(img, 0, 0, svgWidth, svgHeight);\n canvas.toBlob(blob => {\n resolve(blob);\n }, 'image/png', 1.0);\n } catch (err) {\n reject(err);\n }\n };\n \n img.onerror = reject;\n // Use data URI instead of blob URL to avoid tainting the canvas\n const svgDataUrl = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(modifiedSvgString)}`;\n img.src = svgDataUrl;\n });\n}\n\n/**\n * Rasterize a GeoJSON Leaflet map to PNG data URL (following Gem's guide)\n * @param {HTMLElement} liveContainer - The live map container element\n * @returns {Promise<string|null>} PNG data URL or null if failed\n */\nasync function rasterizeGeoJSONMap(liveContainer) {\n try {\n const map = liveContainer._map;\n if (!map) {\n console.warn('No map found on container');\n return null;\n }\n \n // Get container dimensions\n const mapRect = liveContainer.getBoundingClientRect();\n const width = Math.round(mapRect.width);\n const height = Math.round(mapRect.height);\n \n if (width === 0 || height === 0) {\n console.warn('Map container has zero dimensions');\n return null;\n }\n \n // Create canvas sized to the map container\n const canvas = document.createElement('canvas');\n const dpr = window.devicePixelRatio || 1;\n \n // Set canvas size with DPR for sharpness\n canvas.width = width * dpr;\n canvas.height = height * dpr;\n canvas.style.width = width + 'px';\n canvas.style.height = height + 'px';\n \n const ctx = canvas.getContext('2d');\n ctx.scale(dpr, dpr);\n \n // White background\n ctx.fillStyle = '#FFFFFF';\n ctx.fillRect(0, 0, width, height);\n \n // 1. Draw tiles from THIS container only\n const tiles = liveContainer.querySelectorAll('.leaflet-tile');\n \n const tilePromises = [];\n for (const tile of tiles) {\n tilePromises.push(new Promise((resolve) => {\n const img = new Image();\n img.crossOrigin = 'anonymous';\n \n img.onload = () => {\n try {\n // Calculate tile position relative to container\n const tileRect = tile.getBoundingClientRect();\n const offsetX = tileRect.left - mapRect.left;\n const offsetY = tileRect.top - mapRect.top;\n \n // Draw tile at correct position\n ctx.drawImage(img, offsetX, offsetY, tileRect.width, tileRect.height);\n } catch (err) {\n console.warn('Failed to draw tile:', err);\n }\n resolve();\n };\n \n img.onerror = () => {\n console.warn('Failed to load tile:', tile.src);\n resolve();\n };\n \n img.src = tile.src;\n }));\n }\n \n // Wait for all tiles to load\n await Promise.all(tilePromises);\n \n // 2. Draw vector overlays (SVG paths for GeoJSON features)\n const svgOverlays = liveContainer.querySelectorAll('svg:not(.leaflet-attribution-flag)');\n \n for (const svg of svgOverlays) {\n // Skip attribution/control overlays\n if (svg.closest('.leaflet-control')) continue;\n \n try {\n const svgRect = svg.getBoundingClientRect();\n const offsetX = svgRect.left - mapRect.left;\n const offsetY = svgRect.top - mapRect.top;\n \n // Serialize SVG\n const serializer = new XMLSerializer();\n const svgStr = serializer.serializeToString(svg);\n const svgBlob = new Blob([svgStr], { type: 'image/svg+xml;charset=utf-8' });\n const url = URL.createObjectURL(svgBlob);\n \n // Draw SVG overlay\n await new Promise((resolve, reject) => {\n const img = new Image();\n img.onload = () => {\n ctx.drawImage(img, offsetX, offsetY, svgRect.width, svgRect.height);\n URL.revokeObjectURL(url);\n resolve();\n };\n img.onerror = () => {\n URL.revokeObjectURL(url);\n reject(new Error('Failed to load SVG overlay'));\n };\n img.src = url;\n });\n } catch (err) {\n console.warn('Failed to draw SVG overlay:', err);\n }\n }\n \n // 3. Draw marker icons if any\n const markerIcons = liveContainer.querySelectorAll('.leaflet-marker-icon');\n \n for (const marker of markerIcons) {\n try {\n const img = new Image();\n img.crossOrigin = 'anonymous';\n \n await new Promise((resolve) => {\n img.onload = () => {\n const markerRect = marker.getBoundingClientRect();\n const offsetX = markerRect.left - mapRect.left;\n const offsetY = markerRect.top - mapRect.top;\n ctx.drawImage(img, offsetX, offsetY, markerRect.width, markerRect.height);\n resolve();\n };\n img.onerror = resolve;\n img.src = marker.src;\n });\n } catch (err) {\n console.warn('Failed to draw marker icon:', err);\n }\n }\n \n // Return PNG data URL\n return canvas.toDataURL('image/png', 1.0);\n \n } catch (error) {\n console.error('Failed to rasterize GeoJSON map:', error);\n return null;\n }\n}\n\n/**\n * Get rendered content as rich HTML suitable for clipboard\n * @param {HTMLElement} previewPanel - The preview panel element to copy from\n * @returns {Promise<{success: boolean, html?: string, text?: string}>}\n */\nexport async function getRenderedContent(previewPanel) {\n if (!previewPanel) {\n throw new Error('No preview panel available');\n }\n \n // Check if MathJax needs to render (only if not already rendered)\n const mathBlocks = previewPanel.querySelectorAll('.math-display');\n if (mathBlocks.length > 0) {\n // Check if already rendered (has mjx-container inside)\n const needsRendering = Array.from(mathBlocks).some(block => !block.querySelector('mjx-container'));\n \n if (needsRendering && window.MathJax && window.MathJax.typesetPromise) {\n try {\n await window.MathJax.typesetPromise(Array.from(mathBlocks));\n } catch (err) {\n console.warn('MathJax typesetting failed:', err);\n }\n } else {\n }\n }\n \n // Clone the preview panel to avoid modifying the actual DOM\n const clone = previewPanel.cloneNode(true);\n \n // Process different fence types for rich copy\n try {\n // Phase 1: Process basic markdown elements with inline styles\n \n // 1.1 Text formatting - add inline styles\n clone.querySelectorAll('strong, b').forEach(el => {\n el.style.fontWeight = 'bold';\n });\n \n clone.querySelectorAll('em, i').forEach(el => {\n el.style.fontStyle = 'italic';\n });\n \n clone.querySelectorAll('del, s, strike').forEach(el => {\n el.style.textDecoration = 'line-through';\n });\n \n clone.querySelectorAll('u').forEach(el => {\n el.style.textDecoration = 'underline';\n });\n \n clone.querySelectorAll('code:not(pre code)').forEach(el => {\n el.style.backgroundColor = '#f4f4f4';\n el.style.padding = '2px 4px';\n el.style.borderRadius = '3px';\n el.style.fontFamily = 'monospace';\n el.style.fontSize = '0.9em';\n });\n \n // 1.2 Block elements - add inline styles\n clone.querySelectorAll('h1').forEach(el => {\n el.style.fontSize = '2em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '0.67em';\n el.style.marginBottom = '0.67em';\n });\n \n clone.querySelectorAll('h2').forEach(el => {\n el.style.fontSize = '1.5em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '0.83em';\n el.style.marginBottom = '0.83em';\n });\n \n clone.querySelectorAll('h3').forEach(el => {\n el.style.fontSize = '1.17em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '1em';\n el.style.marginBottom = '1em';\n });\n \n clone.querySelectorAll('h4').forEach(el => {\n el.style.fontSize = '1em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '1.33em';\n el.style.marginBottom = '1.33em';\n });\n \n clone.querySelectorAll('h5').forEach(el => {\n el.style.fontSize = '0.83em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '1.67em';\n el.style.marginBottom = '1.67em';\n });\n \n clone.querySelectorAll('h6').forEach(el => {\n el.style.fontSize = '0.67em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '2.33em';\n el.style.marginBottom = '2.33em';\n });\n \n clone.querySelectorAll('blockquote').forEach(el => {\n el.style.borderLeft = '4px solid #ddd';\n el.style.marginLeft = '0';\n el.style.paddingLeft = '1em';\n el.style.color = '#666';\n });\n \n clone.querySelectorAll('hr').forEach(el => {\n el.style.border = 'none';\n el.style.borderTop = '1px solid #ccc';\n el.style.margin = '1em 0';\n });\n \n // 1.3 Tables - add inline styles\n clone.querySelectorAll('table').forEach(table => {\n table.style.borderCollapse = 'collapse';\n table.style.width = '100%';\n table.style.marginBottom = '1em';\n });\n \n clone.querySelectorAll('th').forEach(th => {\n th.style.border = '1px solid #ccc';\n th.style.padding = '8px';\n th.style.textAlign = 'left';\n th.style.backgroundColor = '#f0f0f0';\n th.style.fontWeight = 'bold';\n });\n \n clone.querySelectorAll('td').forEach(td => {\n td.style.border = '1px solid #ccc';\n td.style.padding = '8px';\n td.style.textAlign = 'left';\n });\n \n // 1.4 Links - add inline styles\n clone.querySelectorAll('a').forEach(a => {\n a.style.color = '#0066cc';\n a.style.textDecoration = 'underline';\n });\n \n // Process code blocks - wrap in table and add syntax highlighting colors\n clone.querySelectorAll('pre code').forEach(block => {\n const pre = block.parentElement;\n \n // Add inline styles for syntax highlighting (GitHub theme colors)\n if (block.classList.contains('hljs')) {\n // Apply inline styles to all highlight.js elements\n block.querySelectorAll('.hljs-keyword').forEach(el => {\n el.style.color = '#d73a49';\n el.style.fontWeight = 'bold';\n });\n block.querySelectorAll('.hljs-string').forEach(el => {\n el.style.color = '#032f62';\n });\n block.querySelectorAll('.hljs-number').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-comment').forEach(el => {\n el.style.color = '#6a737d';\n el.style.fontStyle = 'italic';\n });\n block.querySelectorAll('.hljs-function').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-class').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-title').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-built_in').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-literal').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-meta').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-attr').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-variable').forEach(el => {\n el.style.color = '#e36209';\n });\n block.querySelectorAll('.hljs-regexp').forEach(el => {\n el.style.color = '#032f62';\n });\n block.querySelectorAll('.hljs-selector-class').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-selector-id').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-selector-tag').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-tag').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-name').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-attribute').forEach(el => {\n el.style.color = '#6f42c1';\n });\n }\n \n const table = document.createElement('table');\n table.style.width = '100%';\n table.style.borderCollapse = 'collapse';\n table.style.border = 'none';\n table.style.marginBottom = '1em';\n \n const tr = document.createElement('tr');\n const td = document.createElement('td');\n td.style.backgroundColor = '#f7f7f7';\n td.style.padding = '12px';\n td.style.fontFamily = 'Consolas, Monaco, \"Courier New\", monospace';\n td.style.fontSize = '14px';\n td.style.lineHeight = '1.4';\n td.style.whiteSpace = 'pre';\n td.style.overflowX = 'auto';\n td.style.border = '1px solid #ddd';\n td.style.borderRadius = '4px';\n \n // Move the formatted code content with inline styles\n td.innerHTML = block.innerHTML;\n \n tr.appendChild(td);\n table.appendChild(tr);\n \n // Replace the pre element with the table\n pre.parentNode.replaceChild(table, pre);\n });\n \n // Process images - convert to data URLs and ensure proper dimensions\n const images = clone.querySelectorAll('img');\n for (const img of images) {\n // Ensure image has dimensions for Google Docs compatibility\n if (!img.width && img.naturalWidth) {\n img.width = img.naturalWidth;\n }\n if (!img.height && img.naturalHeight) {\n img.height = img.naturalHeight;\n }\n \n // Set max dimensions to prevent huge images\n const maxWidth = 800;\n const maxHeight = 600;\n if (img.width > maxWidth || img.height > maxHeight) {\n const scale = Math.min(maxWidth / img.width, maxHeight / img.height);\n img.width = Math.round(img.width * scale);\n img.height = Math.round(img.height * scale);\n }\n \n // Ensure width and height attributes are set\n if (img.width) {\n img.setAttribute('width', img.width.toString());\n img.style.width = img.width + 'px';\n }\n if (img.height) {\n img.setAttribute('height', img.height.toString());\n img.style.height = img.height + 'px';\n }\n \n // Add v:shapes for Word compatibility\n if (!img.getAttribute('v:shapes')) {\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n }\n \n // Skip if already a data URL\n if (img.src && !img.src.startsWith('data:')) {\n try {\n // Try to convert to data URL\n const response = await fetch(img.src);\n const blob = await response.blob();\n \n // Check if image is too large (Google Docs has limits)\n const maxSize = 2 * 1024 * 1024; // 2MB limit for inline images\n if (blob.size > maxSize) {\n console.warn('Image too large for inline data URL:', img.src, 'Size:', blob.size);\n // For large images, we might want to resize or keep the URL\n continue;\n }\n \n const dataUrl = await new Promise(resolve => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result);\n reader.readAsDataURL(blob);\n });\n img.src = dataUrl;\n } catch (err) {\n console.warn('Failed to convert image to data URL:', img.src, err);\n // Keep original src if conversion fails\n }\n }\n }\n \n // Phase 2: Process fence block types\n // 1. Process STL 3D models - convert canvas to image or placeholder\n const stlContainers = clone.querySelectorAll('.qde-stl-container');\n for (const container of stlContainers) {\n try {\n // Find the corresponding original container to get the canvas\n const containerId = container.dataset.stlId;\n const originalContainer = previewPanel.querySelector(`.qde-stl-container[data-stl-id=\"${containerId}\"]`);\n \n if (originalContainer) {\n // Look for canvas element in the original container (Three.js WebGL canvas)\n const canvas = originalContainer.querySelector('canvas');\n if (canvas && canvas.width > 0 && canvas.height > 0) {\n try {\n // Get Three.js references stored on the container (like squibview)\n const renderer = originalContainer._threeRenderer;\n const scene = originalContainer._threeScene;\n const camera = originalContainer._threeCamera;\n \n // If we have access to the Three.js objects, force render the scene\n if (renderer && scene && camera) {\n renderer.render(scene, camera);\n }\n \n // Try to capture the canvas as an image\n const dataUrl = canvas.toDataURL('image/png', 1.0);\n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Use canvas dimensions for the image\n const imgWidth = canvas.width / 2; // Divide by scale factor (2x for retina)\n const imgHeight = canvas.height / 2;\n \n // Set both HTML attributes and CSS properties for maximum compatibility\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none';\n img.style.maxHeight = 'none';\n img.style.border = '1px solid #ddd';\n img.style.borderRadius = '4px';\n img.style.margin = '0.5em 0';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'STL 3D Model';\n \n container.parentNode.replaceChild(img, container);\n continue;\n } catch (canvasErr) {\n console.warn('Failed to convert STL canvas to image (likely WebGL context issue):', canvasErr);\n }\n } else {\n console.warn('No valid canvas found in STL container');\n }\n } else {\n console.warn('Could not find original STL container');\n }\n } catch (err) {\n console.error('Error processing STL container for copy:', err);\n }\n \n // Fallback to placeholder if canvas conversion fails\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[STL 3D Model - Interactive content not available in copy]';\n container.parentNode.replaceChild(placeholder, container);\n }\n \n // 2. Process Mermaid diagrams - convert SVG to PNG\n const mermaidContainers = clone.querySelectorAll('.mermaid');\n for (const container of mermaidContainers) {\n const svg = container.querySelector('svg');\n if (svg) {\n try {\n const pngBlob = await svgToPng(svg);\n const dataUrl = await new Promise(resolve => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result);\n reader.readAsDataURL(pngBlob);\n });\n \n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Use the exact same dimension calculation logic as svgToPng (like squibview)\n const isMermaidSvg = svg.closest('.mermaid') || svg.classList.contains('mermaid');\n const hasExplicitDimensions = svg.getAttribute('width') && svg.getAttribute('height');\n \n let imgWidth, imgHeight;\n \n if (isMermaidSvg || !hasExplicitDimensions) {\n // For Mermaid or other generated SVGs, prioritize computed dimensions\n imgWidth = svg.clientWidth || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n parseFloat(svg.getAttribute('width')) || 400;\n imgHeight = svg.clientHeight || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n parseFloat(svg.getAttribute('height')) || 300;\n } else {\n // For explicit SVGs (like fenced SVG blocks), prioritize explicit attributes\n imgWidth = parseFloat(svg.getAttribute('width')) || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n svg.clientWidth || 400;\n imgHeight = parseFloat(svg.getAttribute('height')) || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n svg.clientHeight || 300;\n }\n \n // Set both HTML attributes and CSS properties for maximum compatibility (like squibview)\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none'; // Prevent CSS from constraining the image\n img.style.maxHeight = 'none';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'Mermaid Diagram';\n \n container.parentNode.replaceChild(img, container);\n } catch (err) {\n console.warn('Failed to convert Mermaid diagram:', err);\n // Fallback: leave as SVG\n }\n }\n }\n \n // 3. Process Chart.js charts - convert canvas to image\n const chartContainers = clone.querySelectorAll('.qde-chart-container');\n for (const container of chartContainers) {\n try {\n const containerId = container.dataset.chartId;\n const originalContainer = previewPanel.querySelector(`.qde-chart-container[data-chart-id=\"${containerId}\"]`);\n \n if (originalContainer) {\n const canvas = originalContainer.querySelector('canvas');\n if (canvas && canvas.width > 0 && canvas.height > 0) {\n try {\n const dataUrl = canvas.toDataURL('image/png', 1.0);\n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Use canvas dimensions for the image\n const imgWidth = canvas.width;\n const imgHeight = canvas.height;\n \n // Set both HTML attributes and CSS properties for maximum compatibility\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none';\n img.style.maxHeight = 'none';\n img.style.margin = '0.5em 0';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'Chart';\n \n container.parentNode.replaceChild(img, container);\n continue;\n } catch (canvasErr) {\n console.warn('Failed to convert chart canvas to image:', canvasErr);\n }\n }\n }\n } catch (err) {\n console.warn('Error processing chart container:', err);\n }\n \n // Fallback to placeholder\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[Chart - Interactive content not available in copy]';\n container.parentNode.replaceChild(placeholder, container);\n }\n \n // 4. Process SVG fenced blocks - convert to PNG\n const svgContainers = clone.querySelectorAll('.qde-svg-container svg');\n for (const svg of svgContainers) {\n try {\n const pngBlob = await svgToPng(svg);\n const dataUrl = await new Promise(resolve => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result);\n reader.readAsDataURL(pngBlob);\n });\n \n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Calculate dimensions for the SVG\n const hasExplicitDimensions = svg.getAttribute('width') && svg.getAttribute('height');\n \n let imgWidth, imgHeight;\n \n if (hasExplicitDimensions) {\n // For explicit SVGs (like fenced SVG blocks), prioritize explicit attributes\n imgWidth = parseFloat(svg.getAttribute('width')) || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n svg.clientWidth || 400;\n imgHeight = parseFloat(svg.getAttribute('height')) || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n svg.clientHeight || 300;\n } else {\n // For generated SVGs, prioritize computed dimensions\n imgWidth = svg.clientWidth || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n parseFloat(svg.getAttribute('width')) || 400;\n imgHeight = svg.clientHeight || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n parseFloat(svg.getAttribute('height')) || 300;\n }\n \n // Set both HTML attributes and CSS properties for maximum compatibility\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none'; // Prevent CSS from constraining the image\n img.style.maxHeight = 'none';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'SVG Image';\n \n svg.parentNode.replaceChild(img, svg);\n } catch (err) {\n console.warn('Failed to convert SVG to image:', err);\n // Leave as SVG if conversion fails\n }\n }\n \n // 5. Process Math equations - convert to PNG images (exactly like SquibView)\n const mathElements = Array.from(clone.querySelectorAll('.math-display'));\n \n if (mathElements.length > 0) {\n for (const mathEl of mathElements) {\n try {\n // Find SVG inside the math element (MathJax creates it)\n const svg = mathEl.querySelector('svg');\n if (!svg) {\n console.warn('No SVG found in math element, skipping');\n continue;\n }\n \n // Convert SVG to PNG data URL (exactly like SquibView)\n const serializer = new XMLSerializer();\n const svgStr = serializer.serializeToString(svg);\n const svgBlob = new Blob([svgStr], { type: \"image/svg+xml;charset=utf-8\" });\n const url = URL.createObjectURL(svgBlob);\n \n const img = new Image();\n const dataUrl = await new Promise((resolve, reject) => {\n img.onload = function () {\n const canvas = document.createElement('canvas');\n \n // Determine SVG dimensions robustly (exactly like SquibView)\n let width, height;\n try {\n // First try baseVal.value (works for absolute units)\n width = svg.width.baseVal.value;\n height = svg.height.baseVal.value;\n } catch (e) {\n // Fallback for relative units - use viewBox or rendered size\n if (svg.viewBox && svg.viewBox.baseVal) {\n width = svg.viewBox.baseVal.width;\n height = svg.viewBox.baseVal.height;\n } else {\n // Use the natural size of the loaded image\n width = img.naturalWidth || img.width || 200;\n height = img.naturalHeight || img.height || 50;\n }\n }\n \n // Scale down for much smaller paste sizes\n const targetMaxWidth = 150; // Further reduced\n const targetMaxHeight = 45; // Further reduced\n \n // Apply aggressive downsizing for MathJax SVGs\n let scaleFactor = 0.04; // Further reduced for smaller output\n \n let scaledWidth = width * scaleFactor;\n let scaledHeight = height * scaleFactor;\n \n // If still too large after base scaling, scale down further\n if (scaledWidth > targetMaxWidth || scaledHeight > targetMaxHeight) {\n const scaleX = targetMaxWidth / scaledWidth;\n const scaleY = targetMaxHeight / scaledHeight;\n scaleFactor *= Math.min(scaleX, scaleY);\n }\n \n width *= scaleFactor;\n height *= scaleFactor;\n \n // Use higher DPR for crisp rendering at smaller sizes\n const dpr = 2; // Fixed 2x for consistent quality\n canvas.width = width * dpr;\n canvas.height = height * dpr;\n canvas.style.width = width + 'px';\n canvas.style.height = height + 'px';\n \n const ctx = canvas.getContext('2d');\n ctx.scale(dpr, dpr);\n \n // White background\n ctx.fillStyle = \"#FFFFFF\";\n ctx.fillRect(0, 0, width, height);\n \n // Draw the SVG image at logical size\n ctx.drawImage(img, 0, 0, width, height);\n \n // Clean up URL\n URL.revokeObjectURL(url);\n \n // Return data URL\n resolve(canvas.toDataURL('image/png'));\n };\n \n img.onerror = () => {\n URL.revokeObjectURL(url);\n reject(new Error('Failed to load SVG image'));\n };\n \n img.src = url;\n });\n \n // Replace math element with img tag containing the PNG data URL\n const imgElement = document.createElement('img');\n imgElement.src = dataUrl;\n \n // Extract dimensions from the data URL canvas\n const img2 = new Image();\n img2.src = dataUrl;\n await new Promise((resolve) => {\n img2.onload = resolve;\n img2.onerror = resolve;\n setTimeout(resolve, 100); // Timeout fallback\n });\n \n // Set explicit dimensions (accounting for DPR)\n const displayWidth = img2.naturalWidth / 2; // Divide by DPR\n const displayHeight = img2.naturalHeight / 2;\n \n imgElement.width = displayWidth;\n imgElement.height = displayHeight;\n imgElement.style.cssText = `display:inline-block;margin:0.5em 0;width:${displayWidth}px;height:${displayHeight}px;vertical-align:middle;`;\n imgElement.alt = 'Math equation';\n \n mathEl.parentNode.replaceChild(imgElement, mathEl);\n } catch (error) {\n console.error('Failed to convert math element to image:', error);\n // Keep the original element if conversion fails\n }\n }\n }\n \n // 2. Process GeoJSON maps - convert to static images (following Gem's guide)\n const geojsonContainers = clone.querySelectorAll('.geojson-container');\n if (geojsonContainers.length > 0) {\n \n for (const clonedContainer of geojsonContainers) {\n try {\n // Find the corresponding live container by matching data-original-source\n const originalSource = clonedContainer.getAttribute('data-original-source');\n if (!originalSource) {\n console.warn('No original source found for GeoJSON container');\n continue;\n }\n \n // Find live container with same source\n let liveContainer = null;\n const allLiveContainers = previewPanel.querySelectorAll('.geojson-container');\n for (const candidate of allLiveContainers) {\n if (candidate.getAttribute('data-original-source') === originalSource) {\n liveContainer = candidate;\n break;\n }\n }\n \n if (!liveContainer) {\n console.warn('Could not find live GeoJSON container');\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Interactive content not available in copy]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n continue;\n }\n \n // Check if map is ready\n const map = liveContainer._map;\n if (!map) {\n console.warn('Map not initialized yet');\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Still loading]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n continue;\n }\n \n // Rasterize the map to PNG\n const dataUrl = await rasterizeGeoJSONMap(liveContainer);\n \n if (dataUrl) {\n // Replace with image\n const img = document.createElement('img');\n img.src = dataUrl;\n img.style.cssText = 'width: 100%; height: 300px; border: 1px solid #ddd; border-radius: 4px; margin: 0.5em 0;';\n img.alt = 'GeoJSON Map';\n clonedContainer.parentNode.replaceChild(img, clonedContainer);\n } else {\n // Fallback placeholder\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Interactive content not available in copy]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n }\n \n } catch (error) {\n console.error('Failed to process GeoJSON container:', error);\n // Replace with placeholder\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Interactive content not available in copy]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n }\n }\n }\n \n \n \n // 6. Process GeoJSON/Leaflet maps - capture as single image (compose tiles + overlays)\n const mapContainers = clone.querySelectorAll('[data-qd-lang=\"geojson\"]');\n for (const container of mapContainers) {\n try {\n const containerId = container.id;\n const originalContainer = containerId ? previewPanel.querySelector(`#${containerId}`) : null;\n if (!originalContainer) continue;\n const leafletContainer = originalContainer.querySelector('.leaflet-container');\n if (!leafletContainer) continue;\n\n const dpr = Math.max(1, window.devicePixelRatio || 1);\n const width = leafletContainer.clientWidth || 600;\n const height = leafletContainer.clientHeight || 400;\n const canvas = document.createElement('canvas');\n canvas.width = Math.round(width * dpr);\n canvas.height = Math.round(height * dpr);\n const ctx = canvas.getContext('2d');\n ctx.scale(dpr, dpr);\n ctx.fillStyle = '#FFFFFF';\n ctx.fillRect(0, 0, width, height);\n\n const leafRect = leafletContainer.getBoundingClientRect();\n\n // Draw tiles (snap to integer pixels to avoid seams)\n const tiles = Array.from(leafletContainer.querySelectorAll('img.leaflet-tile'));\n for (const tile of tiles) {\n try {\n const r = tile.getBoundingClientRect();\n const x = Math.round(r.left - leafRect.left);\n const y = Math.round(r.top - leafRect.top);\n const w = Math.round(r.width);\n const h = Math.round(r.height);\n const overlaps = !(r.right <= leafRect.left || r.left >= leafRect.right || r.bottom <= leafRect.top || r.top >= leafRect.bottom);\n const style = window.getComputedStyle(tile);\n if (w > 0 && h > 0 && overlaps && style.display !== 'none' && style.visibility !== 'hidden') {\n ctx.drawImage(tile, x, y, w + 1, h + 1);\n }\n } catch (e) {\n console.warn('Failed to draw tile:', e);\n }\n }\n\n // Draw SVG overlays (paths, markers)\n const overlaySvgs = originalContainer.querySelectorAll('.leaflet-overlay-pane svg');\n for (const svg of overlaySvgs) {\n try {\n const svgStr = new XMLSerializer().serializeToString(svg);\n const dataUrl = 'data:image/svg+xml;charset=utf-8,' + encodeURIComponent(svgStr);\n const img = new Image();\n await new Promise((resolve) => { img.onload = resolve; img.onerror = resolve; img.src = dataUrl; });\n const r = svg.getBoundingClientRect();\n const x = Math.round(r.left - leafRect.left);\n const y = Math.round(r.top - leafRect.top);\n const w = Math.round(r.width);\n const h = Math.round(r.height);\n const overlaps = !(r.right <= leafRect.left || r.left >= leafRect.right || r.bottom <= leafRect.top || r.top >= leafRect.bottom);\n if (w > 0 && h > 0 && overlaps) ctx.drawImage(img, x, y, w, h);\n } catch (e) {\n console.warn('Failed to draw overlay SVG:', e);\n }\n }\n\n // Draw marker icons (PNG/SVG img elements)\n const markerIcons = originalContainer.querySelectorAll('.leaflet-marker-pane img.leaflet-marker-icon');\n for (const icon of markerIcons) {\n try {\n const r = icon.getBoundingClientRect();\n const x = Math.round(r.left - leafRect.left);\n const y = Math.round(r.top - leafRect.top);\n const w = Math.round(r.width);\n const h = Math.round(r.height);\n const overlaps = !(r.right <= leafRect.left || r.left >= leafRect.right || r.bottom <= leafRect.top || r.top >= leafRect.bottom);\n const style = window.getComputedStyle(icon);\n if (w > 0 && h > 0 && overlaps && style.display !== 'none' && style.visibility !== 'hidden') {\n ctx.drawImage(icon, x, y, w, h);\n }\n } catch (e) {\n console.warn('Failed to draw marker icon:', e);\n }\n }\n\n // Try to produce a data URL (may fail if canvas tainted by CORS tiles)\n let mapDataUrl = '';\n try {\n mapDataUrl = canvas.toDataURL('image/png', 1.0);\n } catch (e) {\n console.warn('Map canvas tainted; falling back to placeholder');\n }\n\n const img = document.createElement('img');\n if (mapDataUrl) {\n img.src = mapDataUrl;\n img.width = width;\n img.height = height;\n img.setAttribute('width', String(width));\n img.setAttribute('height', String(height));\n img.style.width = width + 'px';\n img.style.height = height + 'px';\n img.style.display = 'block';\n img.style.border = '1px solid #ddd';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'Map';\n } else {\n img.alt = 'Map';\n img.style.width = width + 'px';\n img.style.height = height + 'px';\n img.style.border = '1px solid #ddd';\n img.style.backgroundColor = '#f0f0f0';\n }\n\n container.parentNode.replaceChild(img, container);\n } catch (err) {\n console.warn('Failed to process map container:', err);\n }\n }\n \n // 7. Process HTML fence blocks - render the HTML content and process images\n const htmlContainers = clone.querySelectorAll('.qde-html-container');\n for (const container of htmlContainers) {\n try {\n // Get the original source HTML\n const source = container.getAttribute('data-qd-source');\n \n // Check if there's a pre element (fallback display) or actual HTML content\n const pre = container.querySelector('pre');\n \n if (source) {\n // Parse the source HTML\n const tempDiv = document.createElement('div');\n tempDiv.innerHTML = source;\n \n // Process all images in the HTML block\n const htmlImages = tempDiv.querySelectorAll('img');\n for (const img of htmlImages) {\n // Preserve original dimensions from HTML attributes\n const widthAttr = img.getAttribute('width');\n const heightAttr = img.getAttribute('height');\n \n if (widthAttr) {\n img.width = parseInt(widthAttr);\n img.style.width = widthAttr.includes('%') ? widthAttr : `${img.width}px`;\n }\n if (heightAttr) {\n img.height = parseInt(heightAttr);\n img.style.height = heightAttr.includes('%') ? heightAttr : `${img.height}px`;\n }\n \n // Convert to data URL using canvas (like squibview)\n if (img.src && !img.src.startsWith('data:')) {\n try {\n // Use canvas to convert image to data URL (avoids CORS issues)\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n \n // Create new image and wait for it to load\n const tempImg = new Image();\n tempImg.crossOrigin = 'anonymous';\n \n await new Promise((resolve, reject) => {\n tempImg.onload = function() {\n \n // Calculate dimensions preserving aspect ratio\n let displayWidth = 0;\n let displayHeight = 0;\n \n // Use the width specified in HTML (e.g. width=\"250\")\n if (widthAttr && !widthAttr.includes('%')) {\n displayWidth = parseInt(widthAttr);\n }\n \n // Use the height if specified\n if (heightAttr && !heightAttr.includes('%')) {\n displayHeight = parseInt(heightAttr);\n }\n \n \n // If only width is specified, calculate height based on aspect ratio\n if (displayWidth > 0 && displayHeight === 0) {\n if (tempImg.naturalWidth > 0) {\n const aspectRatio = tempImg.naturalHeight / tempImg.naturalWidth;\n displayHeight = Math.round(displayWidth * aspectRatio);\n }\n }\n // If only height is specified, calculate width based on aspect ratio\n else if (displayHeight > 0 && displayWidth === 0) {\n if (tempImg.naturalHeight > 0) {\n const aspectRatio = tempImg.naturalWidth / tempImg.naturalHeight;\n displayWidth = Math.round(displayHeight * aspectRatio);\n }\n }\n // If neither specified, use natural dimensions\n else if (displayWidth === 0 && displayHeight === 0) {\n displayWidth = tempImg.naturalWidth || 250;\n displayHeight = tempImg.naturalHeight || 200;\n }\n \n \n canvas.width = displayWidth;\n canvas.height = displayHeight;\n \n // Draw image to canvas\n ctx.drawImage(tempImg, 0, 0, displayWidth, displayHeight);\n \n // Convert to data URL\n const dataUrl = canvas.toDataURL('image/png', 1.0);\n \n // Update original image\n img.src = dataUrl;\n img.width = displayWidth;\n img.height = displayHeight;\n img.setAttribute('width', displayWidth.toString());\n img.setAttribute('height', displayHeight.toString());\n img.style.width = displayWidth + 'px';\n img.style.height = displayHeight + 'px';\n \n resolve();\n };\n \n tempImg.onerror = function() {\n console.warn('Failed to load HTML fence image:', img.src);\n reject(new Error('Image load failed'));\n };\n \n // Set source - resolve relative paths\n if (img.src.startsWith('http') || img.src.startsWith('//')) {\n tempImg.src = img.src;\n } else {\n // Relative path - let browser resolve it\n const absoluteImg = new Image();\n absoluteImg.src = img.src;\n tempImg.src = absoluteImg.src;\n }\n });\n } catch (err) {\n console.warn('Failed to convert HTML fence image:', img.src, err);\n }\n }\n \n // Add v:shapes for Word compatibility\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n }\n \n // Replace container content with processed HTML (whether it had pre or not)\n container.innerHTML = tempDiv.innerHTML;\n } else if (!pre) {\n // Container has rendered HTML already, process its images directly\n const htmlImages = container.querySelectorAll('img');\n for (const img of htmlImages) {\n // Same image processing as above\n const widthAttr = img.getAttribute('width');\n const heightAttr = img.getAttribute('height');\n \n if (widthAttr) {\n img.width = parseInt(widthAttr);\n img.style.width = widthAttr.includes('%') ? widthAttr : `${img.width}px`;\n }\n if (heightAttr) {\n img.height = parseInt(heightAttr);\n img.style.height = heightAttr.includes('%') ? heightAttr : `${img.height}px`;\n }\n \n if (img.src && !img.src.startsWith('data:')) {\n try {\n // Use same canvas approach as above\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n const tempImg = new Image();\n tempImg.crossOrigin = 'anonymous';\n \n await new Promise((resolve, reject) => {\n tempImg.onload = function() {\n // Calculate dimensions preserving aspect ratio\n let displayWidth = img.width || 0;\n let displayHeight = img.height || 0;\n \n // If only width is specified, calculate height based on aspect ratio\n if (displayWidth && !displayHeight) {\n const aspectRatio = tempImg.naturalHeight / tempImg.naturalWidth;\n displayHeight = Math.round(displayWidth * aspectRatio);\n }\n // If only height is specified, calculate width based on aspect ratio\n else if (displayHeight && !displayWidth) {\n const aspectRatio = tempImg.naturalWidth / tempImg.naturalHeight;\n displayWidth = Math.round(displayHeight * aspectRatio);\n }\n // If neither specified, use natural dimensions\n else if (!displayWidth && !displayHeight) {\n displayWidth = tempImg.naturalWidth || 250;\n displayHeight = tempImg.naturalHeight || Math.round(250 * (tempImg.naturalHeight / tempImg.naturalWidth));\n }\n \n canvas.width = displayWidth;\n canvas.height = displayHeight;\n ctx.drawImage(tempImg, 0, 0, displayWidth, displayHeight);\n \n const dataUrl = canvas.toDataURL('image/png', 1.0);\n img.src = dataUrl;\n img.width = displayWidth;\n img.height = displayHeight;\n img.setAttribute('width', displayWidth.toString());\n img.setAttribute('height', displayHeight.toString());\n img.style.width = displayWidth + 'px';\n img.style.height = displayHeight + 'px';\n \n resolve();\n };\n \n tempImg.onerror = function() {\n console.warn('Failed to load HTML fence image:', img.src);\n reject(new Error('Image load failed'));\n };\n \n if (img.src.startsWith('http') || img.src.startsWith('//')) {\n tempImg.src = img.src;\n } else {\n const absoluteImg = new Image();\n absoluteImg.src = img.src;\n tempImg.src = absoluteImg.src;\n }\n });\n } catch (err) {\n console.warn('Failed to convert HTML fence image:', img.src, err);\n }\n }\n \n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n }\n }\n } catch (err) {\n console.warn('Failed to process HTML container:', err);\n }\n }\n \n // 8. Tables are already HTML tables from the built-in renderer\n // No processing needed\n \n // Wrap in proper HTML structure for rich text editors\n const fragment = clone.innerHTML;\n const htmlContent = `\n <!DOCTYPE html>\n <html xmlns:v=\"urn:schemas-microsoft-com:vml\"\n xmlns:o=\"urn:schemas-microsoft-com:office:office\"\n xmlns:w=\"urn:schemas-microsoft-com:office:word\">\n <head>\n <meta charset=\"utf-8\">\n <style>\n /* Table styling */\n table { border-collapse: collapse; width: 100%; margin-bottom: 1em; }\n th, td { border: 1px solid #ccc; padding: 8px; text-align: left; }\n th { background-color: #f0f0f0; font-weight: bold; }\n \n /* Code block styling */\n pre { background-color: #f4f4f4; padding: 1em; border-radius: 4px; overflow-x: auto; }\n code { font-family: monospace; background-color: #f4f4f4; padding: 0.2em 0.4em; border-radius: 3px; }\n \n /* Image handling */\n img { display: block; max-width: 100%; height: auto; margin: 0.5em 0; }\n \n /* Blockquote */\n blockquote { border-left: 4px solid #ddd; margin-left: 0; padding-left: 1em; color: #666; }\n \n /* Math equations centered like squibview */\n .math-display { text-align: center; margin: 1em 0; }\n .math-display img { display: inline-block; margin: 0 auto; }\n </style>\n </head>\n <body><!--StartFragment-->${fragment}<!--EndFragment--></body>\n </html>`;\n \n // Get plain text version\n const text = clone.textContent || clone.innerText || '';\n \n // Get platform for clipboard strategy (like squibview)\n const platform = getPlatform();\n \n if (platform === 'macos') {\n // macOS approach (like squibview)\n try {\n await navigator.clipboard.write([\n new ClipboardItem({\n 'text/html': new Blob([htmlContent], { type: 'text/html' }),\n 'text/plain': new Blob([text], { type: 'text/plain' })\n })\n ]);\n return { success: true, html: htmlContent, text };\n } catch (modernErr) {\n console.warn('Modern clipboard API failed, trying Safari fallback:', modernErr);\n // Safari fallback (selection-based HTML of fragment)\n if (copyToClipboard(fragment)) {\n return { success: true, html: htmlContent, text };\n }\n throw new Error('Fallback copy failed');\n }\n } else {\n // Windows/Linux approach (like squibview)\n const tempDiv = document.createElement('div');\n tempDiv.style.position = 'fixed';\n tempDiv.style.left = '-9999px';\n tempDiv.style.top = '0';\n // Use fragment for selection-based fallback copy\n tempDiv.innerHTML = fragment;\n document.body.appendChild(tempDiv);\n \n try {\n await navigator.clipboard.write([\n new ClipboardItem({\n 'text/html': new Blob([htmlContent], { type: 'text/html' }),\n 'text/plain': new Blob([text], { type: 'text/plain' })\n })\n ]);\n return { success: true, html: htmlContent, text };\n } catch (modernErr) {\n console.warn('Modern clipboard API failed, trying execCommand fallback:', modernErr);\n const selection = window.getSelection();\n const range = document.createRange();\n range.selectNodeContents(tempDiv);\n selection.removeAllRanges();\n selection.addRange(range);\n \n const successful = document.execCommand('copy');\n if (!successful) {\n throw new Error('Fallback copy failed');\n }\n return { success: true, html: htmlContent, text };\n } finally {\n if (tempDiv && tempDiv.parentNode) {\n document.body.removeChild(tempDiv);\n }\n }\n }\n \n } catch (err) {\n console.error('Failed to copy rendered content:', err);\n throw err;\n }\n}\n\n","/**\n * Quikdown Editor - A drop-in markdown editor control\n * @version 1.0.5\n * @license BSD-2-Clause\n */\n\nimport quikdown_bd from './quikdown_bd.js';\nimport { getRenderedContent } from './quikdown_edit_copy.js';\n\n// Default options\nconst DEFAULT_OPTIONS = {\n mode: 'split', // 'source' | 'preview' | 'split'\n showToolbar: true,\n showRemoveHR: false, // Show button to remove horizontal rules (---) \n theme: 'auto', // 'light' | 'dark' | 'auto'\n lazy_linefeeds: false,\n inline_styles: false, // Use CSS classes (false) or inline styles (true)\n debounceDelay: 20, // Reduced from 100ms for better responsiveness\n placeholder: 'Start typing markdown...',\n plugins: {\n highlightjs: false,\n mermaid: false\n },\n customFences: {}, // { 'language': (code, lang) => html }\n enableComplexFences: true // Enable CSV tables, math rendering, SVG, etc.\n};\n\n/**\n * Quikdown Editor - A complete markdown editing solution\n */\nclass QuikdownEditor {\n constructor(container, options = {}) {\n // Resolve container\n this.container = typeof container === 'string' \n ? document.querySelector(container) \n : container;\n \n if (!this.container) {\n throw new Error('QuikdownEditor: Invalid container');\n }\n \n // Merge options\n this.options = { ...DEFAULT_OPTIONS, ...options };\n \n // State\n this._markdown = '';\n this._html = '';\n this.currentMode = this.options.mode;\n this.updateTimer = null;\n \n // Initialize\n this.initPromise = this.init();\n }\n \n /**\n * Initialize the editor\n */\n async init() {\n // Load plugins if requested\n await this.loadPlugins();\n \n // Build UI\n this.buildUI();\n \n // Attach event listeners\n this.attachEvents();\n \n // Apply initial theme\n this.applyTheme();\n \n // Set initial mode\n this.setMode(this.currentMode);\n \n // Set initial content if provided\n if (this.options.initialContent) {\n this.setMarkdown(this.options.initialContent);\n }\n }\n \n /**\n * Build the editor UI\n */\n buildUI() {\n // Clear container\n this.container.innerHTML = '';\n \n // Add editor class\n this.container.classList.add('qde-container');\n \n // Create toolbar if enabled\n if (this.options.showToolbar) {\n this.toolbar = this.createToolbar();\n this.container.appendChild(this.toolbar);\n }\n \n // Create editor area\n this.editorArea = document.createElement('div');\n this.editorArea.className = 'qde-editor';\n \n // Create source panel\n this.sourcePanel = document.createElement('div');\n this.sourcePanel.className = 'qde-source';\n \n this.sourceTextarea = document.createElement('textarea');\n this.sourceTextarea.className = 'qde-textarea';\n this.sourceTextarea.placeholder = this.options.placeholder;\n this.sourcePanel.appendChild(this.sourceTextarea);\n \n // Create preview panel\n this.previewPanel = document.createElement('div');\n this.previewPanel.className = 'qde-preview';\n this.previewPanel.contentEditable = true;\n \n // Add panels to editor\n this.editorArea.appendChild(this.sourcePanel);\n this.editorArea.appendChild(this.previewPanel);\n this.container.appendChild(this.editorArea);\n \n // Add built-in styles if not already present\n this.injectStyles();\n }\n \n /**\n * Create toolbar\n */\n createToolbar() {\n const toolbar = document.createElement('div');\n toolbar.className = 'qde-toolbar';\n \n // Mode buttons\n const modes = ['source', 'split', 'preview'];\n const modeLabels = { source: 'Source', split: 'Split', preview: 'Rendered' };\n modes.forEach(mode => {\n const btn = document.createElement('button');\n btn.className = 'qde-btn';\n btn.dataset.mode = mode;\n btn.textContent = modeLabels[mode];\n btn.title = `Switch to ${modeLabels[mode]} view`;\n toolbar.appendChild(btn);\n });\n \n // Spacer\n const spacer = document.createElement('span');\n spacer.className = 'qde-spacer';\n toolbar.appendChild(spacer);\n \n // Copy buttons\n const copyButtons = [\n { action: 'copy-markdown', text: 'Copy MD', title: 'Copy markdown to clipboard' },\n { action: 'copy-html', text: 'Copy HTML', title: 'Copy HTML to clipboard' },\n { action: 'copy-rendered', text: 'Copy Rendered', title: 'Copy rich text to clipboard' }\n ];\n \n copyButtons.forEach(({ action, text, title }) => {\n const btn = document.createElement('button');\n btn.className = 'qde-btn';\n btn.dataset.action = action;\n btn.textContent = text;\n btn.title = title;\n toolbar.appendChild(btn);\n });\n \n // Remove HR button (if enabled)\n if (this.options.showRemoveHR) {\n const removeHRBtn = document.createElement('button');\n removeHRBtn.className = 'qde-btn';\n removeHRBtn.dataset.action = 'remove-hr';\n removeHRBtn.textContent = 'Remove HR';\n removeHRBtn.title = 'Remove all horizontal rules (---) from markdown';\n toolbar.appendChild(removeHRBtn);\n }\n \n return toolbar;\n }\n \n /**\n * Inject built-in styles\n */\n injectStyles() {\n if (document.getElementById('qde-styles')) return;\n \n const style = document.createElement('style');\n style.id = 'qde-styles';\n style.textContent = `\n .qde-container {\n display: flex;\n flex-direction: column;\n height: 100%;\n border: 1px solid #ddd;\n border-radius: 4px;\n overflow: hidden;\n background: white;\n }\n \n .qde-toolbar {\n display: flex;\n align-items: center;\n padding: 8px;\n background: #f5f5f5;\n border-bottom: 1px solid #ddd;\n gap: 4px;\n }\n \n .qde-btn {\n padding: 6px 12px;\n border: 1px solid #ccc;\n background: white;\n border-radius: 3px;\n cursor: pointer;\n font-size: 14px;\n transition: all 0.2s;\n }\n \n .qde-btn:hover {\n background: #e9e9e9;\n border-color: #999;\n }\n \n .qde-btn.active {\n background: #007bff;\n color: white;\n border-color: #0056b3;\n }\n \n .qde-spacer {\n flex: 1;\n }\n \n .qde-editor {\n display: flex;\n flex: 1;\n overflow: hidden;\n }\n \n .qde-source, .qde-preview {\n flex: 1;\n overflow: auto;\n padding: 16px;\n }\n \n .qde-source {\n border-right: 1px solid #ddd;\n }\n \n .qde-textarea {\n width: 100%;\n height: 100%;\n border: none;\n outline: none;\n resize: none;\n font-family: 'Monaco', 'Courier New', monospace;\n font-size: 14px;\n line-height: 1.5;\n }\n \n .qde-preview {\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n font-size: 16px;\n line-height: 1.6;\n outline: none;\n cursor: text; /* Standard text cursor */\n }\n \n /* Fence-specific styles */\n .qde-svg-container {\n max-width: 100%;\n overflow: auto;\n }\n \n .qde-svg-container svg {\n max-width: 100%;\n height: auto;\n }\n \n .qde-html-container {\n /* HTML containers inherit background */\n margin: 12px 0;\n }\n \n .qde-math-container {\n text-align: center;\n margin: 16px 0;\n overflow-x: auto;\n }\n \n /* All tables in preview (both regular markdown and CSV) */\n .qde-preview table {\n width: 100%;\n border-collapse: collapse;\n margin: 12px 0;\n font-size: 14px;\n }\n \n .qde-preview table th,\n .qde-preview table td {\n border: 1px solid #ddd;\n padding: 8px;\n }\n \n /* Support for alignment classes from quikdown */\n .qde-preview .quikdown-left { text-align: left; }\n .qde-preview .quikdown-center { text-align: center; }\n .qde-preview .quikdown-right { text-align: right; }\n \n .qde-preview table th {\n background: #f5f5f5;\n font-weight: bold;\n }\n \n .qde-preview table tr:nth-child(even) {\n background: #f9f9f9;\n }\n \n /* Specific to CSV-generated tables */\n .qde-data-table {\n /* Can add specific CSV table styles here if needed */\n }\n \n .qde-json {\n /* Let highlight.js handle styling */\n overflow-x: auto;\n }\n \n .qde-error {\n background: #fee;\n border: 1px solid #fcc;\n color: #c00;\n padding: 8px;\n border-radius: 4px;\n font-family: monospace;\n font-size: 12px;\n }\n \n /* Read-only complex fence blocks in preview */\n .qde-preview [contenteditable=\"false\"] {\n cursor: auto; /* Use automatic cursor (arrow for non-text) */\n user-select: text;\n position: relative;\n }\n \n /* Ensure proper cursor for editable text elements */\n .qde-preview p,\n .qde-preview h1,\n .qde-preview h2,\n .qde-preview h3,\n .qde-preview h4,\n .qde-preview h5,\n .qde-preview h6,\n .qde-preview li,\n .qde-preview td,\n .qde-preview th,\n .qde-preview blockquote,\n .qde-preview pre[contenteditable=\"true\"],\n .qde-preview code[contenteditable=\"true\"] {\n cursor: text;\n }\n \n \n /* Non-editable complex renderers */\n .qde-preview .qde-svg-container[contenteditable=\"false\"],\n .qde-preview .qde-html-container[contenteditable=\"false\"],\n .qde-preview .qde-math-container[contenteditable=\"false\"],\n .qde-preview .mermaid[contenteditable=\"false\"] {\n opacity: 0.98;\n }\n \n /* Subtle hover effect for read-only blocks */\n .qde-preview [contenteditable=\"false\"]:hover::after {\n content: \"Read-only\";\n position: absolute;\n top: 2px;\n right: 2px;\n font-size: 10px;\n color: #999;\n background: rgba(255, 255, 255, 0.9);\n padding: 2px 4px;\n border-radius: 2px;\n pointer-events: none;\n }\n \n /* Fix list padding in preview */\n .qde-preview ul,\n .qde-preview ol {\n padding-left: 2em;\n margin: 0.5em 0;\n }\n \n .qde-preview li {\n margin: 0.25em 0;\n }\n \n /* Mode-specific visibility */\n .qde-mode-source .qde-preview { display: none; }\n .qde-mode-source .qde-source { border-right: none; }\n .qde-mode-preview .qde-source { display: none; }\n .qde-mode-split .qde-source,\n .qde-mode-split .qde-preview { display: block; }\n \n /* Dark theme */\n .qde-dark {\n background: #1e1e1e;\n color: #e0e0e0;\n }\n \n .qde-dark .qde-toolbar {\n background: #2d2d2d;\n border-color: #444;\n }\n \n .qde-dark .qde-btn {\n background: #3a3a3a;\n color: #e0e0e0;\n border-color: #555;\n }\n \n .qde-dark .qde-btn:hover {\n background: #4a4a4a;\n }\n \n .qde-dark .qde-source {\n border-color: #444;\n }\n \n .qde-dark .qde-textarea {\n background: #1e1e1e;\n color: #e0e0e0;\n }\n \n .qde-dark .qde-preview {\n background: #1e1e1e;\n color: #e0e0e0;\n }\n \n /* Dark mode table styles */\n .qde-dark .qde-preview table th,\n .qde-dark .qde-preview table td {\n border-color: #3a3a3a;\n }\n \n .qde-dark .qde-preview table th {\n background: #2d2d2d;\n }\n \n .qde-dark .qde-preview table tr:nth-child(even) {\n background: #252525;\n }\n \n /* Mobile responsive */\n @media (max-width: 768px) {\n .qde-mode-split .qde-editor {\n flex-direction: column;\n }\n \n .qde-mode-split .qde-source {\n border-right: none;\n border-bottom: 1px solid #ddd;\n }\n }\n `;\n \n document.head.appendChild(style);\n }\n \n /**\n * Attach event listeners\n */\n attachEvents() {\n // Source textarea input\n this.sourceTextarea.addEventListener('input', () => {\n this.handleSourceInput();\n });\n \n // Preview contenteditable input\n this.previewPanel.addEventListener('input', () => {\n this.handlePreviewInput();\n });\n \n // Toolbar buttons\n if (this.toolbar) {\n this.toolbar.addEventListener('click', (e) => {\n const btn = e.target.closest('.qde-btn');\n if (!btn) return;\n \n if (btn.dataset.mode) {\n this.setMode(btn.dataset.mode);\n } else if (btn.dataset.action) {\n this.handleAction(btn.dataset.action);\n }\n });\n }\n \n // Keyboard shortcuts\n document.addEventListener('keydown', (e) => {\n if (e.ctrlKey || e.metaKey) {\n switch(e.key) {\n case '1':\n e.preventDefault();\n this.setMode('source');\n break;\n case '2':\n e.preventDefault();\n this.setMode('split');\n break;\n case '3':\n e.preventDefault();\n this.setMode('preview');\n break;\n }\n }\n });\n }\n \n /**\n * Handle source textarea input\n */\n handleSourceInput() {\n clearTimeout(this.updateTimer);\n this.updateTimer = setTimeout(() => {\n this.updateFromMarkdown(this.sourceTextarea.value);\n }, this.options.debounceDelay);\n }\n \n /**\n * Handle preview panel input\n */\n handlePreviewInput() {\n clearTimeout(this.updateTimer);\n this.updateTimer = setTimeout(() => {\n this.updateFromHTML();\n }, this.options.debounceDelay);\n }\n \n /**\n * Update from markdown source\n */\n updateFromMarkdown(markdown) {\n this._markdown = markdown || '';\n \n // Show placeholder if empty\n if (!this._markdown.trim()) {\n this._html = '';\n if (this.currentMode !== 'source') {\n this.previewPanel.innerHTML = '<div style=\"color: #999; font-style: italic; padding: 16px;\">Start typing markdown in the source panel...</div>';\n }\n } else {\n this._html = quikdown_bd(markdown, {\n fence_plugin: this.createFencePlugin(),\n lazy_linefeeds: this.options.lazy_linefeeds,\n inline_styles: this.options.inline_styles\n });\n \n // Update preview if visible\n if (this.currentMode !== 'source') {\n this.previewPanel.innerHTML = this._html;\n // Make all fence blocks non-editable\n this.makeFencesNonEditable();\n \n // Process all math elements with MathJax if loaded (like squibview)\n if (window.MathJax && window.MathJax.typesetPromise) {\n const mathElements = this.previewPanel.querySelectorAll('.math-display');\n if (mathElements.length > 0) {\n mathElements.forEach(el => {\n });\n window.MathJax.typesetPromise(Array.from(mathElements))\n .then(() => {\n mathElements.forEach(el => {\n const mjxContainer = el.querySelector('mjx-container');\n });\n })\n .catch(err => {\n console.warn('MathJax batch processing failed:', err);\n });\n }\n }\n }\n }\n \n // Trigger change event\n if (this.options.onChange) {\n this.options.onChange(this._markdown, this._html);\n }\n }\n \n /**\n * Update from HTML preview\n */\n updateFromHTML() {\n // Clone the preview panel to avoid modifying the actual DOM\n const clonedPanel = this.previewPanel.cloneNode(true);\n \n // Pre-process special elements on the clone\n this.preprocessSpecialElements(clonedPanel);\n \n this._html = this.previewPanel.innerHTML;\n this._markdown = quikdown_bd.toMarkdown(clonedPanel, {\n fence_plugin: this.createFencePlugin()\n });\n \n // Update source if visible\n if (this.currentMode !== 'preview') {\n this.sourceTextarea.value = this._markdown;\n }\n \n // Trigger change event\n if (this.options.onChange) {\n this.options.onChange(this._markdown, this._html);\n }\n }\n \n /**\n * Pre-process special elements before markdown conversion\n */\n preprocessSpecialElements(panel) {\n if (!panel) return;\n \n // Restore non-editable complex fences from their data attributes\n const complexFences = panel.querySelectorAll('[contenteditable=\"false\"][data-qd-source]');\n complexFences.forEach(element => {\n const source = element.getAttribute('data-qd-source');\n const fence = element.getAttribute('data-qd-fence') || '```';\n const lang = element.getAttribute('data-qd-lang') || '';\n \n // Create a pre element with the original source\n const pre = document.createElement('pre');\n pre.setAttribute('data-qd-fence', fence);\n if (lang) pre.setAttribute('data-qd-lang', lang);\n const code = document.createElement('code');\n // The source is already the original unescaped content when using setAttribute\n // No need to unescape since browser handles it automatically\n code.textContent = source;\n pre.appendChild(code);\n \n // Replace the complex element with pre\n element.parentNode.replaceChild(pre, element);\n });\n \n // Convert CSV tables back to CSV fence blocks (these ARE editable)\n const csvTables = panel.querySelectorAll('table.qde-csv-table[data-qd-lang]');\n csvTables.forEach(table => {\n const lang = table.getAttribute('data-qd-lang');\n if (!lang || !['csv', 'psv', 'tsv'].includes(lang)) return;\n \n const delimiter = lang === 'csv' ? ',' : lang === 'psv' ? '|' : '\\t';\n \n // Extract data from table\n let csv = '';\n \n // Get headers\n const headers = [];\n const headerCells = table.querySelectorAll('thead th');\n headerCells.forEach(th => {\n const text = th.textContent.trim();\n // Quote if contains delimiter or quotes\n const needsQuoting = text.includes(delimiter) || text.includes('\"') || text.includes('\\n');\n headers.push(needsQuoting ? `\"${text.replace(/\"/g, '\"\"')}\"` : text);\n });\n csv += headers.join(delimiter) + '\\n';\n \n // Get rows\n const rows = table.querySelectorAll('tbody tr');\n rows.forEach(tr => {\n const cells = [];\n tr.querySelectorAll('td').forEach(td => {\n const text = td.textContent.trim();\n const needsQuoting = text.includes(delimiter) || text.includes('\"') || text.includes('\\n');\n cells.push(needsQuoting ? `\"${text.replace(/\"/g, '\"\"')}\"` : text);\n });\n csv += cells.join(delimiter) + '\\n';\n });\n \n // Create a pre element with the CSV data\n const pre = document.createElement('pre');\n pre.setAttribute('data-qd-fence', '```');\n pre.setAttribute('data-qd-lang', lang);\n const code = document.createElement('code');\n code.textContent = csv.trim();\n pre.appendChild(code);\n \n // Replace table with pre\n table.parentNode.replaceChild(pre, table);\n });\n }\n \n /**\n * Create fence plugin for syntax highlighting\n */\n createFencePlugin() {\n const render = (code, lang) => {\n // Check custom fences first (they take precedence)\n if (this.options.customFences && this.options.customFences[lang]) {\n try {\n return this.options.customFences[lang](code, lang);\n } catch (err) {\n console.error(`Custom fence plugin error for ${lang}:`, err);\n return `<pre><code class=\"language-${lang}\">${this.escapeHtml(code)}</code></pre>`;\n }\n }\n \n // For bidirectional editing, only apply syntax highlighting\n // Skip complex transformations that break round-trip conversion\n const skipComplexRendering = !this.options.enableComplexFences;\n \n if (!skipComplexRendering) {\n // Built-in lazy loading fence handlers (disabled for now)\n switch(lang) {\n case 'svg':\n return this.renderSVG(code);\n \n case 'html':\n return this.renderHTML(code);\n \n case 'math':\n case 'tex':\n case 'latex':\n return this.renderMath(code, lang);\n \n case 'csv':\n case 'psv':\n case 'tsv':\n return this.renderTable(code, lang);\n \n case 'json':\n case 'json5':\n return this.renderJSON(code, lang);\n \n case 'katex': // Use MathJax for katex fence blocks (backward compatibility)\n return this.renderMath(code, 'katex');\n \n case 'mermaid':\n if (window.mermaid) {\n return this.renderMermaid(code);\n }\n break;\n \n case 'geojson':\n return this.renderGeoJSON(code);\n \n case 'stl':\n return this.renderSTL(code);\n }\n }\n \n // Syntax highlighting support - keep editable for bidirectional\n if (window.hljs && lang && hljs.getLanguage(lang)) {\n const highlighted = hljs.highlight(code, { language: lang }).value;\n // Don't add contenteditable=\"false\" - the bidirectional system can extract text from the highlighted code\n return `<pre data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\"><code class=\"hljs language-${lang}\">${highlighted}</code></pre>`;\n }\n \n // Default: let quikdown handle it\n return undefined;\n };\n \n // Reverse function to extract raw source from rendered HTML\n const reverse = (element) => {\n // Get the language from data attribute\n const lang = element.getAttribute('data-qd-lang') || '';\n let content = '';\n \n // For syntax-highlighted code, extract the raw text\n if (element.querySelector('code.hljs')) {\n const code = element.querySelector('code.hljs');\n content = code.textContent || code.innerText || '';\n }\n // For other code blocks, just get the text content\n else if (element.querySelector('code')) {\n const codeEl = element.querySelector('code');\n content = codeEl.textContent || codeEl.innerText || '';\n }\n // Fallback to element text\n else {\n content = element.textContent || element.innerText || '';\n }\n \n // Return in the format quikdown_bd expects\n return {\n content: content,\n lang: lang,\n fence: '```'\n };\n };\n \n // Return object format for v1.1.0 API with both render and reverse\n return { render, reverse };\n }\n \n /**\n * Render SVG content\n */\n renderSVG(code) {\n try {\n // Basic SVG validation\n const parser = new DOMParser();\n const doc = parser.parseFromString(code, 'image/svg+xml');\n const parseError = doc.querySelector('parsererror');\n \n if (parseError) {\n throw new Error('Invalid SVG');\n }\n \n // Sanitize SVG by removing script tags and event handlers\n const svg = doc.documentElement;\n svg.querySelectorAll('script').forEach(el => el.remove());\n \n // Remove event handlers\n const walker = document.createTreeWalker(svg, NodeFilter.SHOW_ELEMENT);\n let node;\n while (node = walker.nextNode()) {\n for (let i = node.attributes.length - 1; i >= 0; i--) {\n const attr = node.attributes[i];\n if (attr.name.startsWith('on') || attr.value.includes('javascript:')) {\n node.removeAttribute(attr.name);\n }\n }\n }\n \n // Create container element programmatically to avoid attribute escaping issues\n const container = document.createElement('div');\n container.className = 'qde-svg-container';\n container.contentEditable = 'false';\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'svg');\n container.setAttribute('data-qd-source', code); // No escaping needed when using setAttribute!\n container.innerHTML = new XMLSerializer().serializeToString(svg);\n \n // Return the HTML string\n return container.outerHTML;\n } catch (err) {\n const errorContainer = document.createElement('pre');\n errorContainer.className = 'qde-error';\n errorContainer.contentEditable = 'false';\n errorContainer.setAttribute('data-qd-fence', '```');\n errorContainer.setAttribute('data-qd-lang', 'svg');\n errorContainer.textContent = `Invalid SVG: ${err.message}`;\n return errorContainer.outerHTML;\n }\n }\n \n /**\n * Render HTML content with DOMPurify if available\n */\n renderHTML(code) {\n const id = `html-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n \n // If DOMPurify is loaded, use it\n if (window.DOMPurify) {\n const clean = DOMPurify.sanitize(code);\n \n // Create container programmatically\n const container = document.createElement('div');\n container.className = 'qde-html-container';\n container.contentEditable = 'false';\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'html');\n container.setAttribute('data-qd-source', code);\n container.innerHTML = clean;\n \n return container.outerHTML;\n }\n \n // Try to lazy load DOMPurify\n this.lazyLoadLibrary(\n 'DOMPurify',\n () => window.DOMPurify,\n 'https://unpkg.com/dompurify/dist/purify.min.js'\n ).then(loaded => {\n if (loaded) {\n const element = document.getElementById(id);\n if (element) {\n const clean = DOMPurify.sanitize(code);\n element.innerHTML = clean;\n // Update attributes after loading\n element.setAttribute('data-qd-source', code);\n element.setAttribute('data-qd-fence', '```');\n element.setAttribute('data-qd-lang', 'html');\n }\n }\n });\n \n // Return placeholder with bidirectional attributes - non-editable\n const placeholder = document.createElement('div');\n placeholder.id = id;\n placeholder.className = 'qde-html-container';\n placeholder.contentEditable = 'false';\n placeholder.setAttribute('data-qd-fence', '```');\n placeholder.setAttribute('data-qd-lang', 'html');\n placeholder.setAttribute('data-qd-source', code);\n const pre = document.createElement('pre');\n pre.textContent = code;\n placeholder.appendChild(pre);\n \n return placeholder.outerHTML;\n }\n \n /**\n * Render math with MathJax (SVG output for better copy support)\n */\n renderMath(code, lang) {\n const id = `math-${Math.random().toString(36).substring(2, 15)}`;\n \n // Create container exactly like squibview\n const container = document.createElement('div');\n container.id = id;\n container.className = 'math-display';\n container.contentEditable = 'false';\n container.setAttribute('data-source-type', 'math');\n \n // Format content for MathJax (display mode with $$) - exactly like squibview\n const singleLineContent = code.replace(/\\r?\\n/g, ' ').replace(/\\s+/g, ' ').trim();\n container.textContent = `$$${singleLineContent}$$`;\n \n // Add centering style\n container.style.textAlign = 'center';\n container.style.margin = '1em 0';\n \n \n // Ensure MathJax will be loaded (if not already)\n if (!window.MathJax || !window.MathJax.typesetPromise) {\n this.ensureMathJaxLoaded();\n }\n \n // MathJax will be processed in batch after preview update\n return container.outerHTML;\n }\n \n /**\n * Ensures MathJax is loaded (but doesn't process elements)\n */\n ensureMathJaxLoaded() {\n if (typeof window.MathJax === 'undefined' && !window.mathJaxLoading) {\n window.mathJaxLoading = true;\n \n // Configure MathJax before loading\n if (!window.MathJax) {\n window.MathJax = {\n loader: { load: ['input/tex', 'output/svg'] },\n tex: { \n packages: { '[+]': ['ams'] },\n inlineMath: [['$', '$'], ['\\\\(', '\\\\)']],\n displayMath: [['$$', '$$'], ['\\\\[', '\\\\]']],\n processEscapes: true,\n processEnvironments: true\n },\n options: {\n renderActions: { addMenu: [] },\n ignoreHtmlClass: 'tex2jax_ignore',\n processHtmlClass: 'tex2jax_process'\n },\n svg: {\n fontCache: 'none' // Important: self-contained SVGs for copy\n },\n startup: { typeset: false }\n };\n }\n \n const script = document.createElement('script');\n script.src = 'https://cdn.jsdelivr.net/npm/mathjax@3.2.2/es5/tex-svg.js';\n script.async = true;\n script.onload = () => {\n window.mathJaxLoading = false;\n \n // Process any existing math elements (like squibview)\n if (window.MathJax && window.MathJax.typesetPromise) {\n const mathElements = document.querySelectorAll('.math-display');\n if (mathElements.length > 0) {\n window.MathJax.typesetPromise(Array.from(mathElements)).catch(err => {\n console.warn('Initial MathJax processing failed:', err);\n });\n }\n }\n };\n script.onerror = () => {\n window.mathJaxLoading = false;\n console.error('Failed to load MathJax');\n };\n document.head.appendChild(script);\n }\n }\n \n /**\n * Render CSV/PSV/TSV as HTML table\n */\n renderTable(code, lang) {\n const escapedCode = this.escapeHtml(code);\n try {\n const delimiter = lang === 'csv' ? ',' : lang === 'psv' ? '|' : '\\t';\n const lines = code.trim().split('\\n');\n \n if (lines.length === 0) {\n return `<pre data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\" data-qd-source=\"${escapedCode}\">${escapedCode}</pre>`;\n }\n \n // CSV tables CAN be editable - we'll convert HTML table back to CSV\n // Don't need data-qd-source since we convert the table structure back to CSV\n let html = `<table class=\"qde-data-table qde-csv-table\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\">`;\n \n // Parse header\n const header = this.parseCSVLine(lines[0], delimiter);\n html += '<thead><tr>';\n header.forEach(cell => {\n html += `<th>${this.escapeHtml(cell.trim())}</th>`;\n });\n html += '</tr></thead>';\n \n // Parse body\n if (lines.length > 1) {\n html += '<tbody>';\n for (let i = 1; i < lines.length; i++) {\n const row = this.parseCSVLine(lines[i], delimiter);\n html += '<tr>';\n row.forEach(cell => {\n html += `<td>${this.escapeHtml(cell.trim())}</td>`;\n });\n html += '</tr>';\n }\n html += '</tbody>';\n }\n \n html += '</table>';\n return html;\n } catch (err) {\n return `<pre data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\" data-qd-source=\"${escapedCode}\">${escapedCode}</pre>`;\n }\n }\n \n /**\n * Parse CSV line handling quoted values\n */\n parseCSVLine(line, delimiter) {\n const result = [];\n let current = '';\n let inQuotes = false;\n \n for (let i = 0; i < line.length; i++) {\n const char = line[i];\n const nextChar = line[i + 1];\n \n if (char === '\"') {\n if (inQuotes && nextChar === '\"') {\n current += '\"';\n i++; // Skip next quote\n } else {\n inQuotes = !inQuotes;\n }\n } else if (char === delimiter && !inQuotes) {\n result.push(current);\n current = '';\n } else {\n current += char;\n }\n }\n \n result.push(current);\n return result;\n }\n \n /**\n * Render JSON with syntax highlighting\n */\n renderJSON(code, lang) {\n // If highlight.js is available, use it for all JSON\n if (window.hljs && hljs.getLanguage('json')) {\n try {\n // Try to format if valid JSON\n let toHighlight = code;\n try {\n const data = JSON.parse(code);\n toHighlight = JSON.stringify(data, null, 2);\n } catch (e) {\n // Use original if not valid JSON\n }\n \n const highlighted = hljs.highlight(toHighlight, { language: 'json' }).value;\n return `<pre class=\"qde-json\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\"><code class=\"hljs language-json\">${highlighted}</code></pre>`;\n } catch (e) {\n // Fall through if highlighting fails\n }\n }\n \n // No highlighting available - return plain\n return `<pre class=\"qde-json\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\">${this.escapeHtml(code)}</pre>`;\n }\n \n /**\n * Render GeoJSON map\n */\n renderGeoJSON(code) {\n // Generate unique map ID (following SquibView pattern)\n const mapId = `map-${Math.random().toString(36).substr(2, 15)}`;\n \n // Function to render the map\n const renderMap = () => {\n const container = document.getElementById(mapId + '-container');\n if (!container || !window.L) return;\n \n try {\n const data = JSON.parse(code);\n \n // Clear container and set deterministic size for rasterization\n const mapDiv = document.createElement('div');\n mapDiv.id = mapId;\n mapDiv.style.cssText = 'width: 100%; height: 300px;';\n container.innerHTML = '';\n container.appendChild(mapDiv);\n \n // Create the map\n const map = L.map(mapId);\n \n // Store back-reference for capture (per Gem's guide)\n container._map = map; // Avoid window pollution\n \n // Add tile layer with CORS support\n const tileLayer = L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {\n attribution: '',\n crossOrigin: 'anonymous' // Important for canvas capture\n });\n tileLayer.addTo(map);\n \n // Add GeoJSON layer\n const geoJsonLayer = L.geoJSON(data);\n geoJsonLayer.addTo(map);\n \n // Fit bounds if valid\n if (geoJsonLayer.getBounds().isValid()) {\n map.fitBounds(geoJsonLayer.getBounds());\n } else {\n map.setView([0, 0], 2);\n }\n \n // Store references for copy-time capture\n container._tileLayer = tileLayer;\n container._geoJsonLayer = geoJsonLayer;\n \n // Optional: Wait for tiles to load for better capture\n tileLayer.on('load', () => {\n container.setAttribute('data-tiles-loaded', 'true');\n });\n \n } catch (err) {\n container.innerHTML = `<pre class=\"qde-error\">GeoJSON error: ${this.escapeHtml(err.message)}</pre>`;\n }\n };\n \n // Check if Leaflet is already loaded\n if (window.L) {\n // Render after DOM update\n setTimeout(renderMap, 0);\n } else {\n // Lazy load Leaflet only if not already loading\n if (!window._qde_leaflet_loading) {\n window._qde_leaflet_loading = this.lazyLoadLibrary(\n 'Leaflet',\n () => window.L,\n 'https://unpkg.com/leaflet@1.9.4/dist/leaflet.js',\n 'https://unpkg.com/leaflet@1.9.4/dist/leaflet.css'\n ).catch(err => {\n console.warn('Failed to load Leaflet:', err);\n // Clear the loading promise so it can be retried\n window._qde_leaflet_loading = null;\n return false;\n });\n }\n \n window._qde_leaflet_loading.then(loaded => {\n if (loaded) {\n renderMap();\n } else {\n const element = document.getElementById(id);\n if (element) {\n element.innerHTML = '<div style=\"padding: 20px; text-align: center; color: #666;\">Failed to load map library</div>';\n }\n }\n }).catch(() => {\n // Error already handled above\n });\n }\n \n // Return container following SquibView pattern\n const container = document.createElement('div');\n container.className = 'geojson-container';\n container.id = mapId + '-container';\n container.style.cssText = 'width: 100%; height: 300px; border: 1px solid #ddd; border-radius: 4px; margin: 0.5em 0; background: #f0f0f0;';\n container.contentEditable = 'false';\n \n // Preserve source for copy-time identification (per Gem's guide)\n container.setAttribute('data-source-type', 'geojson');\n container.setAttribute('data-original-source', this.escapeHtml(code));\n \n // For bidirectional editing\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'geojson');\n container.setAttribute('data-qd-source', code);\n \n container.textContent = 'Loading map...';\n \n return container.outerHTML;\n }\n \n /**\n * Render STL 3D model\n */\n renderSTL(code) {\n const id = `qde-stl-viewer-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n \n // Function to render the 3D model\n const render3D = () => {\n const element = document.getElementById(id);\n if (!element) return;\n \n // Check if Three.js is available\n if (typeof window.THREE === 'undefined') {\n element.innerHTML = '<div style=\"padding: 20px; text-align: center; color: #666;\">Three.js library not loaded. Add <script src=\"https://unpkg.com/three@0.147.0/build/three.min.js\"></script> to your HTML.</div>';\n return;\n }\n \n try {\n const THREE = window.THREE;\n \n // Create scene\n const scene = new THREE.Scene();\n scene.background = new THREE.Color(0xf0f0f0);\n \n // Create camera\n const camera = new THREE.PerspectiveCamera(75, element.clientWidth / 400, 0.1, 1000);\n \n // Create renderer\n const renderer = new THREE.WebGLRenderer({ antialias: true });\n renderer.setSize(element.clientWidth, 400);\n element.innerHTML = '';\n element.appendChild(renderer.domElement);\n \n // Store Three.js references for copy functionality (like squibview)\n element._threeScene = scene;\n element._threeCamera = camera;\n element._threeRenderer = renderer;\n \n // Parse STL data (ASCII format)\n const geometry = this.parseSTL(code);\n const material = new THREE.MeshLambertMaterial({ color: 0x0066ff });\n const mesh = new THREE.Mesh(geometry, material);\n scene.add(mesh);\n \n // Add lighting\n const ambientLight = new THREE.AmbientLight(0x404040, 0.6);\n scene.add(ambientLight);\n \n const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);\n directionalLight.position.set(1, 1, 1).normalize();\n scene.add(directionalLight);\n \n // Position camera based on object bounds\n const box = new THREE.Box3().setFromObject(mesh);\n const center = box.getCenter(new THREE.Vector3());\n const size = box.getSize(new THREE.Vector3());\n const maxDim = Math.max(size.x, size.y, size.z);\n \n camera.position.set(center.x + maxDim, center.y + maxDim, center.z + maxDim);\n camera.lookAt(center);\n \n // Animate\n const animate = () => {\n requestAnimationFrame(animate);\n mesh.rotation.y += 0.01;\n renderer.render(scene, camera);\n };\n animate();\n } catch (err) {\n console.error('STL rendering error:', err);\n element.innerHTML = `<pre class=\"qde-error\">STL error: ${this.escapeHtml(err.message)}</pre>`;\n }\n };\n \n // Render after DOM update\n setTimeout(render3D, 0);\n \n // Return placeholder with data-stl-id for copy functionality\n return `<div id=\"${id}\" class=\"qde-stl-container\" data-stl-id=\"${id}\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"stl\" data-qd-source=\"${this.escapeHtml(code)}\" contenteditable=\"false\" style=\"height: 400px; background: #f0f0f0; display: flex; align-items: center; justify-content: center;\">Loading 3D model...</div>`;\n }\n \n /**\n * Parse ASCII STL format\n * @param {string} stlData - The STL file content\n * @returns {THREE.BufferGeometry} - The parsed geometry\n */\n parseSTL(stlData) {\n const THREE = window.THREE;\n const geometry = new THREE.BufferGeometry();\n const vertices = [];\n const normals = [];\n \n const lines = stlData.split('\\n');\n let currentNormal = null;\n \n for (let line of lines) {\n line = line.trim();\n \n if (line.startsWith('facet normal')) {\n const parts = line.split(/\\s+/);\n currentNormal = [parseFloat(parts[2]), parseFloat(parts[3]), parseFloat(parts[4])];\n } else if (line.startsWith('vertex')) {\n const parts = line.split(/\\s+/);\n vertices.push(parseFloat(parts[1]), parseFloat(parts[2]), parseFloat(parts[3]));\n if (currentNormal) {\n normals.push(currentNormal[0], currentNormal[1], currentNormal[2]);\n }\n }\n }\n \n geometry.setAttribute('position', new THREE.Float32BufferAttribute(vertices, 3));\n geometry.setAttribute('normal', new THREE.Float32BufferAttribute(normals, 3));\n \n return geometry;\n }\n \n /**\n * Render Mermaid diagram\n */\n renderMermaid(code) {\n const id = `mermaid-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n setTimeout(() => {\n const element = document.getElementById(id);\n if (element && window.mermaid) {\n mermaid.render(id + '-svg', code).then(result => {\n element.innerHTML = result.svg;\n }).catch(err => {\n element.innerHTML = `<pre>Error rendering diagram: ${err.message}</pre>`;\n });\n }\n }, 0);\n \n // Create container programmatically\n const container = document.createElement('div');\n container.id = id;\n container.className = 'mermaid';\n container.contentEditable = 'false';\n container.setAttribute('data-qd-source', code);\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'mermaid');\n container.textContent = 'Loading diagram...';\n \n return container.outerHTML;\n }\n \n /**\n * Escape HTML for attributes\n */\n escapeHtml(text) {\n return (text ?? \"\").replace(/[&\"'<>]/g, m => \n ({'&':'&','\"':'"',\"'\":''','<':'<','>':'>'}[m]));\n }\n \n /**\n * Make complex fence blocks non-editable\n */\n makeFencesNonEditable() {\n if (!this.previewPanel) return;\n \n // Only make specific complex fence types non-editable\n // SVG, HTML, Math, Mermaid already have contenteditable=\"false\" set\n // Syntax-highlighted code also has it set\n \n // Don't make regular code blocks or tables non-editable\n // They can be edited and properly round-trip\n }\n \n /**\n * Load plugins dynamically\n */\n async loadPlugins() {\n const promises = [];\n \n // Load highlight.js (check if already loaded)\n if (this.options.plugins.highlightjs && !window.hljs) {\n promises.push(\n this.loadScript('https://unpkg.com/@highlightjs/cdn-assets/highlight.min.js'),\n this.loadCSS('https://unpkg.com/@highlightjs/cdn-assets/styles/github.min.css')\n );\n }\n \n // Load mermaid (check if already loaded)\n if (this.options.plugins.mermaid && !window.mermaid) {\n promises.push(\n this.loadScript('https://unpkg.com/mermaid/dist/mermaid.min.js').then(() => {\n if (window.mermaid) {\n mermaid.initialize({ startOnLoad: false });\n }\n })\n );\n }\n \n await Promise.all(promises);\n }\n \n /**\n * Lazy load library if not already loaded\n */\n async lazyLoadLibrary(name, check, scriptUrl, cssUrl = null) {\n // Check if library is already loaded\n if (check()) {\n return true;\n }\n \n try {\n const promises = [];\n \n // Load script\n if (scriptUrl) {\n promises.push(this.loadScript(scriptUrl));\n }\n \n // Load CSS if provided\n if (cssUrl) {\n promises.push(this.loadCSS(cssUrl));\n }\n \n await Promise.all(promises);\n \n // Verify library loaded\n return check();\n } catch (err) {\n console.error(`Failed to load ${name}:`, err);\n return false;\n }\n }\n \n /**\n * Load external script\n */\n loadScript(src) {\n return new Promise((resolve, reject) => {\n const script = document.createElement('script');\n script.src = src;\n script.onload = resolve;\n script.onerror = reject;\n document.head.appendChild(script);\n });\n }\n \n /**\n * Load external CSS\n */\n loadCSS(href) {\n return new Promise((resolve) => {\n const link = document.createElement('link');\n link.rel = 'stylesheet';\n link.href = href;\n link.onload = resolve;\n document.head.appendChild(link);\n // Resolve anyway after timeout (CSS doesn't always fire onload)\n setTimeout(resolve, 1000);\n });\n }\n \n /**\n * Apply theme\n */\n applyTheme() {\n const theme = this.options.theme;\n \n if (theme === 'auto') {\n // Check system preference\n const isDark = window.matchMedia('(prefers-color-scheme: dark)').matches;\n this.container.classList.toggle('qde-dark', isDark);\n \n // Listen for changes\n window.matchMedia('(prefers-color-scheme: dark)').addEventListener('change', (e) => {\n this.container.classList.toggle('qde-dark', e.matches);\n });\n } else {\n this.container.classList.toggle('qde-dark', theme === 'dark');\n }\n }\n \n /**\n * Set lazy linefeeds option\n * @param {boolean} enabled - Whether to enable lazy linefeeds\n */\n setLazyLinefeeds(enabled) {\n this.options.lazy_linefeeds = enabled;\n // Re-render if we have content\n if (this._markdown) {\n this.updateFromMarkdown(this._markdown);\n }\n }\n \n /**\n * Get lazy linefeeds option\n * @returns {boolean}\n */\n getLazyLinefeeds() {\n return this.options.lazy_linefeeds;\n }\n \n /**\n * Set debounce delay for input updates\n * @param {number} delay - Delay in milliseconds (0 for instant)\n */\n setDebounceDelay(delay) {\n this.options.debounceDelay = Math.max(0, delay);\n }\n \n /**\n * Get current debounce delay\n * @returns {number} Delay in milliseconds\n */\n getDebounceDelay() {\n return this.options.debounceDelay;\n }\n \n /**\n * Set editor mode\n */\n setMode(mode) {\n if (!['source', 'preview', 'split'].includes(mode)) return;\n \n this.currentMode = mode;\n this.container.className = `qde-container qde-mode-${mode}`;\n \n // Update toolbar buttons\n if (this.toolbar) {\n this.toolbar.querySelectorAll('.qde-btn[data-mode]').forEach(btn => {\n btn.classList.toggle('active', btn.dataset.mode === mode);\n });\n }\n \n // Apply theme class\n if (this.container.classList.contains('qde-dark')) {\n this.container.classList.add('qde-dark');\n }\n \n // Make fence blocks non-editable when showing preview\n if (mode !== 'source') {\n setTimeout(() => this.makeFencesNonEditable(), 0);\n }\n \n // Trigger mode change event\n if (this.options.onModeChange) {\n this.options.onModeChange(mode);\n }\n }\n \n /**\n * Handle toolbar actions\n */\n handleAction(action) {\n switch(action) {\n case 'copy-markdown':\n this.copy('markdown');\n break;\n case 'copy-html':\n this.copy('html');\n break;\n case 'copy-rendered':\n this.copyRendered();\n break;\n case 'remove-hr':\n this.removeHR();\n break;\n }\n }\n \n /**\n * Copy content to clipboard\n */\n async copy(type) {\n const content = type === 'markdown' ? this._markdown : this._html;\n \n try {\n await navigator.clipboard.writeText(content);\n \n // Visual feedback\n const btn = this.toolbar.querySelector(`[data-action=\"copy-${type}\"]`);\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Copied!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n } catch (err) {\n console.error('Failed to copy:', err);\n }\n }\n \n // Public API\n \n /**\n * Get current markdown\n */\n get markdown() {\n return this._markdown;\n }\n \n /**\n * Set markdown content\n */\n set markdown(value) {\n this.setMarkdown(value);\n }\n \n /**\n * Get current HTML\n */\n get html() {\n return this._html;\n }\n \n /**\n * Get current mode\n */\n get mode() {\n return this.currentMode;\n }\n \n /**\n * Set markdown content\n */\n async setMarkdown(markdown) {\n // Wait for initialization if needed\n if (this.initPromise) {\n await this.initPromise;\n }\n \n this._markdown = markdown;\n if (this.sourceTextarea) {\n this.sourceTextarea.value = markdown;\n }\n this.updateFromMarkdown(markdown);\n }\n \n /**\n * Get markdown content\n */\n getMarkdown() {\n return this._markdown;\n }\n \n /**\n * Get HTML content\n */\n getHTML() {\n return this._html;\n }\n \n /**\n * Remove all horizontal rules (---) from markdown\n */\n async removeHR() {\n // Remove standalone HR lines (3 or more dashes/underscores/asterisks)\n // Matches: ---, ___, ***, ----, etc. with optional spaces\n const cleaned = this._markdown\n .split('\\n')\n .filter(line => {\n // Keep lines that aren't just HR patterns\n const trimmed = line.trim();\n // Match HR patterns: 3+ of -, _, or * with optional spaces between\n return !(/^[-_*](\\s*[-_*]){2,}\\s*$/.test(trimmed));\n })\n .join('\\n');\n \n // Update the markdown\n await this.setMarkdown(cleaned);\n \n // Visual feedback if toolbar button exists\n const btn = this.toolbar?.querySelector('[data-action=\"remove-hr\"]');\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Removed!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n }\n \n /**\n * Copy rendered content as rich text\n */\n async copyRendered() {\n try {\n const result = await getRenderedContent(this.previewPanel);\n if (result.success) {\n // Visual feedback\n const btn = this.toolbar?.querySelector('[data-action=\"copy-rendered\"]');\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Copied!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n }\n } catch (err) {\n console.error('Failed to copy rendered content:', err);\n }\n }\n \n /**\n * Destroy the editor\n */\n destroy() {\n // Clear timers\n clearTimeout(this.updateTimer);\n \n // Clear container\n this.container.innerHTML = '';\n this.container.classList.remove('qde-container', 'qde-dark');\n \n // Remove injected styles (only if no other editors exist)\n const otherEditors = document.querySelectorAll('.qde-container');\n if (otherEditors.length === 0) {\n const style = document.getElementById('qde-styles');\n if (style) style.remove();\n }\n }\n}\n\n// Export\nexport default QuikdownEditor;\n\n// Export for CommonJS (needed for bundled ESM to work with Jest)\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = QuikdownEditor;\n}\n\n// Also export for UMD builds\nif (typeof window !== 'undefined') {\n window.QuikdownEditor = QuikdownEditor;\n}"],"names":["CLASS_PREFIX","PLACEHOLDER_CB","ESC_MAP","QUIKDOWN_STYLES","h1","h2","h3","h4","h5","h6","pre","code","blockquote","table","th","td","hr","img","a","strong","em","del","ul","ol","li","quikdown","markdown","options","fence_plugin","inline_styles","bidirectional","lazy_linefeeds","getAttr","styles","tag","additionalStyle","style","includes","replace","trim","endsWith","classAttr","createGetAttr","escapeHtml","text","m","dataQd","marker","sanitizeUrl","url","allowUnsafe","trimmedUrl","lowerUrl","toLowerCase","dangerousProtocols","protocol","startsWith","html","codeBlocks","inlineCodes","match","fence","lang","placeholder","length","langTrimmed","render","push","trimEnd","custom","hasReverse","reverse","lines","split","result","inTable","tableLines","i","line","test","tableHtml","buildTable","join","processTable","hashes","content","level","listStack","indent","Math","floor","isOrdered","listType","listItemContent","taskListClass","taskMatch","checked","taskContent","isChecked","list","pop","type","currentList","liAttr","processLists","alt","src","sanitizedSrc","allow_unsafe_urls","altAttr","srcAttr","href","sanitizedHref","rel","textAttr","prefix","sanitizedUrl","forEach","pattern","blocks","bi","b","offset","substring","RegExp","replacement","block","undefined","langClass","codeAttr","langAttr","fenceAttr","processInlineMarkdown","separatorIndex","headerLines","slice","bodyLines","alignments","map","cell","trimmed","alignStyle","processedCell","quikdown_bd","async","svgToPng","svgElement","needsWhiteBackground","Promise","resolve","reject","svgString","XMLSerializer","serializeToString","canvas","document","createElement","ctx","getContext","Image","isMermaidSvg","closest","classList","contains","hasExplicitDimensions","getAttribute","svgWidth","svgHeight","clientWidth","viewBox","baseVal","width","parseFloat","clientHeight","height","modifiedSvgString","tempDiv","innerHTML","tempSvg","querySelector","setAttribute","toString","scale","onload","fillStyle","fillRect","drawImage","toBlob","blob","err","onerror","svgDataUrl","encodeURIComponent","rasterizeGeoJSONMap","liveContainer","_map","console","warn","mapRect","getBoundingClientRect","round","dpr","window","devicePixelRatio","tiles","querySelectorAll","tilePromises","tile","crossOrigin","tileRect","offsetX","left","offsetY","top","all","svgOverlays","svg","svgRect","svgStr","svgBlob","Blob","URL","createObjectURL","revokeObjectURL","Error","markerIcons","markerRect","toDataURL","error","getRenderedContent","previewPanel","mathBlocks","Array","from","some","MathJax","typesetPromise","clone","cloneNode","el","fontWeight","fontStyle","textDecoration","backgroundColor","padding","borderRadius","fontFamily","fontSize","marginTop","marginBottom","borderLeft","marginLeft","paddingLeft","color","border","borderTop","margin","borderCollapse","textAlign","parentElement","tr","lineHeight","whiteSpace","overflowX","appendChild","parentNode","replaceChild","images","naturalWidth","naturalHeight","maxWidth","maxHeight","min","random","substr","response","fetch","maxSize","size","dataUrl","reader","FileReader","onloadend","readAsDataURL","stlContainers","container","containerId","dataset","stlId","originalContainer","renderer","_threeRenderer","scene","_threeScene","camera","_threeCamera","imgWidth","imgHeight","canvasErr","cssText","textContent","mermaidContainers","pngBlob","chartContainers","chartId","svgContainers","mathElements","mathEl","value","e","scaleFactor","scaledWidth","scaledHeight","scaleX","scaleY","imgElement","img2","setTimeout","displayWidth","displayHeight","geojsonContainers","clonedContainer","originalSource","allLiveContainers","candidate","mapContainers","id","leafletContainer","max","leafRect","r","x","y","w","h","overlaps","right","bottom","getComputedStyle","display","visibility","overlaySvgs","icon","mapDataUrl","String","htmlContainers","source","htmlImages","widthAttr","heightAttr","parseInt","tempImg","aspectRatio","absoluteImg","fragment","htmlContent","innerText","platform","navigator","userAgent","getPlatform","clipboard","write","ClipboardItem","success","modernErr","position","overflow","body","range","createRange","selectNodeContents","selection","getSelection","removeAllRanges","addRange","execCommand","removeChild","copyToClipboard","emitStyles","theme","themeOverrides","_textColor","css","Object","entries","themedStyle","oldColor","newColor","configure","version","module","exports","keys","key","toMarkdown","htmlOrElement","Element","walkNode","node","parentContext","nodeType","Node","TEXT_NODE","ELEMENT_NODE","tagName","childContent","child","childNodes","parentTag","repeat","boldMarker","emMarker","delMarker","codeMarker","fenceMarker","codeEl","quoteMarker","linkText","walkList","alignData","thead","headerRow","headers","_","align","tbody","row","cells","walkTable","trailingBlankLines","divLang","divFence","divSource","temp","mermaidPre","preSource","sourceElement","mermaidElement","listNode","depth","index","children","checkbox","itemContent","DEFAULT_OPTIONS","mode","showToolbar","showRemoveHR","debounceDelay","plugins","highlightjs","mermaid","customFences","enableComplexFences","QuikdownEditor","constructor","this","_markdown","_html","currentMode","updateTimer","initPromise","init","loadPlugins","buildUI","attachEvents","applyTheme","setMode","initialContent","setMarkdown","add","toolbar","createToolbar","editorArea","className","sourcePanel","sourceTextarea","contentEditable","injectStyles","modeLabels","preview","btn","title","spacer","action","removeHRBtn","getElementById","head","addEventListener","handleSourceInput","handlePreviewInput","target","handleAction","ctrlKey","metaKey","preventDefault","clearTimeout","updateFromMarkdown","updateFromHTML","createFencePlugin","makeFencesNonEditable","then","catch","onChange","clonedPanel","preprocessSpecialElements","panel","element","delimiter","csv","needsQuoting","renderSVG","renderHTML","renderMath","renderTable","renderJSON","renderMermaid","renderGeoJSON","renderSTL","hljs","getLanguage","highlight","language","doc","DOMParser","parseFromString","documentElement","remove","walker","createTreeWalker","NodeFilter","SHOW_ELEMENT","nextNode","attributes","attr","name","removeAttribute","outerHTML","errorContainer","message","Date","now","DOMPurify","clean","sanitize","lazyLoadLibrary","loaded","singleLineContent","ensureMathJaxLoaded","mathJaxLoading","loader","load","tex","packages","inlineMath","displayMath","processEscapes","processEnvironments","renderActions","addMenu","ignoreHtmlClass","processHtmlClass","fontCache","startup","typeset","script","escapedCode","header","parseCSVLine","current","inQuotes","char","nextChar","toHighlight","data","JSON","parse","stringify","mapId","renderMap","L","mapDiv","tileLayer","attribution","addTo","geoJsonLayer","geoJSON","getBounds","isValid","fitBounds","setView","_tileLayer","_geoJsonLayer","on","_qde_leaflet_loading","THREE","Scene","background","Color","PerspectiveCamera","WebGLRenderer","antialias","setSize","domElement","geometry","parseSTL","material","MeshLambertMaterial","mesh","Mesh","ambientLight","AmbientLight","directionalLight","DirectionalLight","set","normalize","box","Box3","setFromObject","center","getCenter","Vector3","getSize","maxDim","z","lookAt","animate","requestAnimationFrame","rotation","stlData","BufferGeometry","vertices","normals","currentNormal","parts","Float32BufferAttribute","promises","loadScript","loadCSS","initialize","startOnLoad","check","scriptUrl","cssUrl","link","isDark","matchMedia","matches","toggle","setLazyLinefeeds","enabled","getLazyLinefeeds","setDebounceDelay","delay","getDebounceDelay","onModeChange","copy","copyRendered","removeHR","writeText","originalText","getMarkdown","getHTML","cleaned","filter","destroy"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"quikdown_edit.umd.min.js","sources":["../src/quikdown.js","../src/quikdown_bd.js","../src/quikdown_edit_copy.js","../src/quikdown_edit.js"],"sourcesContent":["/**\n * quikdown - A minimal markdown parser optimized for chat/LLM output\n * Supports tables, code blocks, lists, and common formatting\n * @param {string} markdown - The markdown source text\n * @param {Object} options - Optional configuration object\n * @param {Function} options.fence_plugin - Custom renderer for fenced code blocks\n * (content, fence_string) => html string\n * @param {boolean} options.inline_styles - If true, uses inline styles instead of classes\n * @param {boolean} options.bidirectional - If true, adds data-qd attributes for source tracking\n * @param {boolean} options.lazy_linefeeds - If true, single newlines become <br> tags\n * @returns {string} - The rendered HTML\n */\n\n// Version will be injected at build time \nconst quikdownVersion = '__QUIKDOWN_VERSION__';\n\n// Constants for reuse\nconst CLASS_PREFIX = 'quikdown-';\nconst PLACEHOLDER_CB = '§CB';\nconst PLACEHOLDER_IC = '§IC';\n\n// Escape map at module level\nconst ESC_MAP = {'&':'&','<':'<','>':'>','\"':'"',\"'\":'''};\n\n// Single source of truth for all style definitions - optimized\nconst QUIKDOWN_STYLES = {\n h1: 'font-size:2em;font-weight:600;margin:.67em 0;text-align:left',\n h2: 'font-size:1.5em;font-weight:600;margin:.83em 0',\n h3: 'font-size:1.25em;font-weight:600;margin:1em 0',\n h4: 'font-size:1em;font-weight:600;margin:1.33em 0',\n h5: 'font-size:.875em;font-weight:600;margin:1.67em 0',\n h6: 'font-size:.85em;font-weight:600;margin:2em 0',\n pre: 'background:#f4f4f4;padding:10px;border-radius:4px;overflow-x:auto;margin:1em 0',\n code: 'background:#f0f0f0;padding:2px 4px;border-radius:3px;font-family:monospace',\n blockquote: 'border-left:4px solid #ddd;margin-left:0;padding-left:1em',\n table: 'border-collapse:collapse;width:100%;margin:1em 0',\n th: 'border:1px solid #ddd;padding:8px;background-color:#f2f2f2;font-weight:bold;text-align:left',\n td: 'border:1px solid #ddd;padding:8px;text-align:left',\n hr: 'border:none;border-top:1px solid #ddd;margin:1em 0',\n img: 'max-width:100%;height:auto',\n a: 'color:#06c;text-decoration:underline',\n strong: 'font-weight:bold',\n em: 'font-style:italic',\n del: 'text-decoration:line-through',\n ul: 'margin:.5em 0;padding-left:2em',\n ol: 'margin:.5em 0;padding-left:2em',\n li: 'margin:.25em 0',\n // Task list specific styles\n 'task-item': 'list-style:none',\n 'task-checkbox': 'margin-right:.5em'\n};\n\n// Factory function to create getAttr for a given context\nfunction createGetAttr(inline_styles, styles) {\n return function(tag, additionalStyle = '') {\n if (inline_styles) {\n let style = styles[tag];\n if (!style && !additionalStyle) return '';\n \n // Remove default text-align if we're adding a different alignment\n if (additionalStyle && additionalStyle.includes('text-align') && style && style.includes('text-align')) {\n style = style.replace(/text-align:[^;]+;?/, '').trim();\n // Ensure trailing semicolon before concatenating additionalStyle.\n // Both short-circuit paths of this guard (empty `style` or\n // already-has-`;`) are defensive and unreachable with the\n // current QUIKDOWN_STYLES values — istanbul ignore next.\n /* istanbul ignore next */\n if (style && !style.endsWith(';')) style += ';';\n }\n \n /* istanbul ignore next - defensive: additionalStyle without style doesn't occur with current tags */\n const fullStyle = additionalStyle ? (style ? `${style}${additionalStyle}` : additionalStyle) : style;\n return ` style=\"${fullStyle}\"`;\n } else {\n const classAttr = ` class=\"${CLASS_PREFIX}${tag}\"`;\n // Apply inline styles for alignment even when using CSS classes\n if (additionalStyle) {\n return `${classAttr} style=\"${additionalStyle}\"`;\n }\n return classAttr;\n }\n };\n}\n\nfunction quikdown(markdown, options = {}) {\n if (!markdown || typeof markdown !== 'string') {\n return '';\n }\n \n const { fence_plugin, inline_styles = false, bidirectional = false, lazy_linefeeds = false } = options;\n const styles = QUIKDOWN_STYLES; // Use module-level styles\n const getAttr = createGetAttr(inline_styles, styles); // Create getAttr once\n\n // Escape HTML entities to prevent XSS\n function escapeHtml(text) {\n return text.replace(/[&<>\"']/g, m => ESC_MAP[m]);\n }\n \n // Helper to add data-qd attributes for bidirectional support.\n // The non-bidirectional branch is a trivial no-op arrow; it's exercised in\n // the core bundle but never in quikdown_bd (which always sets bidirectional=true).\n /* istanbul ignore next - trivial no-op fallback */\n const dataQd = bidirectional ? (marker) => ` data-qd=\"${escapeHtml(marker)}\"` : () => '';\n\n // Sanitize URLs to prevent XSS attacks\n function sanitizeUrl(url, allowUnsafe = false) {\n /* istanbul ignore next - defensive programming, regex ensures url is never empty */\n if (!url) return '';\n \n // If unsafe URLs are explicitly allowed, return as-is\n if (allowUnsafe) return url;\n \n const trimmedUrl = url.trim();\n const lowerUrl = trimmedUrl.toLowerCase();\n \n // Block dangerous protocols\n const dangerousProtocols = ['javascript:', 'vbscript:', 'data:'];\n \n for (const protocol of dangerousProtocols) {\n if (lowerUrl.startsWith(protocol)) {\n // Exception: Allow data:image/* for images\n if (protocol === 'data:' && lowerUrl.startsWith('data:image/')) {\n return trimmedUrl;\n }\n // Return safe empty link for dangerous protocols\n return '#';\n }\n }\n \n return trimmedUrl;\n }\n\n // Process the markdown in phases\n let html = markdown;\n \n // Phase 1: Extract and protect code blocks and inline code\n const codeBlocks = [];\n const inlineCodes = [];\n \n // Extract fenced code blocks first (supports both ``` and ~~~)\n // Match paired fences - ``` with ``` and ~~~ with ~~~\n // Fence must be at start of line\n html = html.replace(/^(```|~~~)([^\\n]*)\\n([\\s\\S]*?)^\\1$/gm, (match, fence, lang, code) => {\n const placeholder = `${PLACEHOLDER_CB}${codeBlocks.length}§`;\n \n // Trim the language specification\n const langTrimmed = lang ? lang.trim() : '';\n \n // If custom fence plugin is provided, use it (v1.1.0: object format required)\n if (fence_plugin && fence_plugin.render && typeof fence_plugin.render === 'function') {\n codeBlocks.push({\n lang: langTrimmed,\n code: code.trimEnd(),\n custom: true,\n fence: fence,\n hasReverse: !!fence_plugin.reverse\n });\n } else {\n codeBlocks.push({\n lang: langTrimmed,\n code: escapeHtml(code.trimEnd()),\n custom: false,\n fence: fence\n });\n }\n return placeholder;\n });\n \n // Extract inline code\n html = html.replace(/`([^`]+)`/g, (match, code) => {\n const placeholder = `${PLACEHOLDER_IC}${inlineCodes.length}§`;\n inlineCodes.push(escapeHtml(code));\n return placeholder;\n });\n \n // Now escape HTML in the rest of the content\n html = escapeHtml(html);\n \n // Phase 2: Process block elements\n \n // Process tables\n html = processTable(html, getAttr);\n \n // Process headings (supports optional trailing #'s)\n html = html.replace(/^(#{1,6})\\s+(.+?)\\s*#*$/gm, (match, hashes, content) => {\n const level = hashes.length;\n return `<h${level}${getAttr('h' + level)}${dataQd(hashes)}>${content}</h${level}>`;\n });\n \n // Process blockquotes (must handle escaped > since we already escaped HTML)\n html = html.replace(/^>\\s+(.+)$/gm, `<blockquote${getAttr('blockquote')}>$1</blockquote>`);\n // Merge consecutive blockquotes\n html = html.replace(/<\\/blockquote>\\n<blockquote>/g, '\\n');\n \n // Process horizontal rules (allow trailing spaces)\n html = html.replace(/^---+\\s*$/gm, `<hr${getAttr('hr')}>`);\n \n // Process lists\n html = processLists(html, getAttr, inline_styles, bidirectional);\n \n // Phase 3: Process inline elements\n \n // Images (must come before links, with URL sanitization)\n html = html.replace(/!\\[([^\\]]*)\\]\\(([^)]+)\\)/g, (match, alt, src) => {\n const sanitizedSrc = sanitizeUrl(src, options.allow_unsafe_urls);\n const altAttr = bidirectional && alt ? ` data-qd-alt=\"${escapeHtml(alt)}\"` : '';\n const srcAttr = bidirectional ? ` data-qd-src=\"${escapeHtml(src)}\"` : '';\n return `<img${getAttr('img')} src=\"${sanitizedSrc}\" alt=\"${alt}\"${altAttr}${srcAttr}${dataQd('!')}>`;\n });\n \n // Links (with URL sanitization)\n html = html.replace(/\\[([^\\]]+)\\]\\(([^)]+)\\)/g, (match, text, href) => {\n // Sanitize URL to prevent XSS\n const sanitizedHref = sanitizeUrl(href, options.allow_unsafe_urls);\n const isExternal = /^https?:\\/\\//i.test(sanitizedHref);\n const rel = isExternal ? ' rel=\"noopener noreferrer\"' : '';\n const textAttr = bidirectional ? ` data-qd-text=\"${escapeHtml(text)}\"` : '';\n return `<a${getAttr('a')} href=\"${sanitizedHref}\"${rel}${textAttr}${dataQd('[')}>${text}</a>`;\n });\n \n // Autolinks - convert bare URLs to clickable links\n html = html.replace(/(^|\\s)(https?:\\/\\/[^\\s<]+)/g, (match, prefix, url) => {\n const sanitizedUrl = sanitizeUrl(url, options.allow_unsafe_urls);\n return `${prefix}<a${getAttr('a')} href=\"${sanitizedUrl}\" rel=\"noopener noreferrer\">${url}</a>`;\n });\n \n // Process inline formatting (bold, italic, strikethrough)\n const inlinePatterns = [\n [/\\*\\*(.+?)\\*\\*/g, 'strong', '**'],\n [/__(.+?)__/g, 'strong', '__'],\n [/(?<!\\*)\\*(?!\\*)(.+?)(?<!\\*)\\*(?!\\*)/g, 'em', '*'],\n [/(?<!_)_(?!_)(.+?)(?<!_)_(?!_)/g, 'em', '_'],\n [/~~(.+?)~~/g, 'del', '~~']\n ];\n \n inlinePatterns.forEach(([pattern, tag, marker]) => {\n html = html.replace(pattern, `<${tag}${getAttr(tag)}${dataQd(marker)}>$1</${tag}>`);\n });\n \n // Line breaks\n if (lazy_linefeeds) {\n // Lazy linefeeds: single newline becomes <br> (except between paragraphs and after/before block elements)\n const blocks = [];\n let bi = 0;\n \n // Protect tables and lists \n html = html.replace(/<(table|[uo]l)[^>]*>[\\s\\S]*?<\\/\\1>/g, m => {\n blocks[bi] = m;\n return `§B${bi++}§`;\n });\n \n // Handle paragraphs and block elements\n html = html.replace(/\\n\\n+/g, '§P§')\n // After block elements\n .replace(/(<\\/(?:h[1-6]|blockquote|pre)>)\\n/g, '$1§N§')\n .replace(/(<(?:h[1-6]|blockquote|pre|hr)[^>]*>)\\n/g, '$1§N§')\n // Before block elements \n .replace(/\\n(<(?:h[1-6]|blockquote|pre|hr)[^>]*>)/g, '§N§$1')\n .replace(/\\n(§B\\d+§)/g, '§N§$1')\n .replace(/(§B\\d+§)\\n/g, '$1§N§')\n // Convert remaining newlines\n .replace(/\\n/g, `<br${getAttr('br')}>`)\n // Restore\n .replace(/§N§/g, '\\n')\n .replace(/§P§/g, '</p><p>');\n \n // Restore protected blocks\n blocks.forEach((b, i) => html = html.replace(`§B${i}§`, b));\n \n html = '<p>' + html + '</p>';\n } else {\n // Standard: two spaces at end of line for line breaks\n html = html.replace(/ {2}$/gm, `<br${getAttr('br')}>`);\n \n // Paragraphs (double newlines)\n // Don't add </p> after block elements (they're not in paragraphs)\n html = html.replace(/\\n\\n+/g, (match, offset) => {\n // Check if we're after a block element closing tag\n const before = html.substring(0, offset);\n if (before.match(/<\\/(h[1-6]|blockquote|ul|ol|table|pre|hr)>$/)) {\n return '<p>'; // Just open a new paragraph\n }\n return '</p><p>'; // Normal paragraph break\n });\n html = '<p>' + html + '</p>';\n }\n \n // Clean up empty paragraphs and unwrap block elements\n const cleanupPatterns = [\n [/<p><\\/p>/g, ''],\n [/<p>(<h[1-6][^>]*>)/g, '$1'],\n [/(<\\/h[1-6]>)<\\/p>/g, '$1'],\n [/<p>(<blockquote[^>]*>)/g, '$1'],\n [/(<\\/blockquote>)<\\/p>/g, '$1'],\n [/<p>(<ul[^>]*>|<ol[^>]*>)/g, '$1'],\n [/(<\\/ul>|<\\/ol>)<\\/p>/g, '$1'],\n [/<p>(<hr[^>]*>)<\\/p>/g, '$1'],\n [/<p>(<table[^>]*>)/g, '$1'],\n [/(<\\/table>)<\\/p>/g, '$1'],\n [/<p>(<pre[^>]*>)/g, '$1'],\n [/(<\\/pre>)<\\/p>/g, '$1'],\n [new RegExp(`<p>(${PLACEHOLDER_CB}\\\\d+§)</p>`, 'g'), '$1']\n ];\n \n cleanupPatterns.forEach(([pattern, replacement]) => {\n html = html.replace(pattern, replacement);\n });\n \n // Fix orphaned closing </p> tags after block elements\n // When a paragraph follows a block element, ensure it has opening <p>\n html = html.replace(/(<\\/(?:h[1-6]|blockquote|ul|ol|table|pre|hr)>)\\n([^<])/g, '$1\\n<p>$2');\n \n // Phase 4: Restore code blocks and inline code\n \n // Restore code blocks\n codeBlocks.forEach((block, i) => {\n let replacement;\n \n if (block.custom && fence_plugin && fence_plugin.render) {\n // Use custom fence plugin (v1.1.0: object format with render function)\n replacement = fence_plugin.render(block.code, block.lang);\n \n // If plugin returns undefined, fall back to default rendering\n if (replacement === undefined) {\n const langClass = !inline_styles && block.lang ? ` class=\"language-${block.lang}\"` : '';\n const codeAttr = inline_styles ? getAttr('code') : langClass;\n const langAttr = bidirectional && block.lang ? ` data-qd-lang=\"${escapeHtml(block.lang)}\"` : '';\n const fenceAttr = bidirectional ? ` data-qd-fence=\"${escapeHtml(block.fence)}\"` : '';\n replacement = `<pre${getAttr('pre')}${fenceAttr}${langAttr}><code${codeAttr}>${escapeHtml(block.code)}</code></pre>`;\n } else if (bidirectional) {\n // If bidirectional and plugin provided HTML, add data attributes for roundtrip\n replacement = replacement.replace(/^<(\\w+)/, \n `<$1 data-qd-fence=\"${escapeHtml(block.fence)}\" data-qd-lang=\"${escapeHtml(block.lang)}\" data-qd-source=\"${escapeHtml(block.code)}\"`);\n }\n } else {\n // Default rendering\n const langClass = !inline_styles && block.lang ? ` class=\"language-${block.lang}\"` : '';\n const codeAttr = inline_styles ? getAttr('code') : langClass;\n const langAttr = bidirectional && block.lang ? ` data-qd-lang=\"${escapeHtml(block.lang)}\"` : '';\n const fenceAttr = bidirectional ? ` data-qd-fence=\"${escapeHtml(block.fence)}\"` : '';\n replacement = `<pre${getAttr('pre')}${fenceAttr}${langAttr}><code${codeAttr}>${block.code}</code></pre>`;\n }\n \n const placeholder = `${PLACEHOLDER_CB}${i}§`;\n html = html.replace(placeholder, replacement);\n });\n \n // Restore inline code\n inlineCodes.forEach((code, i) => {\n const placeholder = `${PLACEHOLDER_IC}${i}§`;\n html = html.replace(placeholder, `<code${getAttr('code')}${dataQd('`')}>${code}</code>`);\n });\n \n return html.trim();\n}\n\n/**\n * Process inline markdown formatting\n */\nfunction processInlineMarkdown(text, getAttr) {\n \n // Process inline formatting patterns\n const patterns = [\n [/\\*\\*(.+?)\\*\\*/g, 'strong'],\n [/__(.+?)__/g, 'strong'],\n [/(?<!\\*)\\*(?!\\*)(.+?)(?<!\\*)\\*(?!\\*)/g, 'em'],\n [/(?<!_)_(?!_)(.+?)(?<!_)_(?!_)/g, 'em'],\n [/~~(.+?)~~/g, 'del'],\n [/`([^`]+)`/g, 'code']\n ];\n \n patterns.forEach(([pattern, tag]) => {\n text = text.replace(pattern, `<${tag}${getAttr(tag)}>$1</${tag}>`);\n });\n \n return text;\n}\n\n/**\n * Process markdown tables\n */\nfunction processTable(text, getAttr) {\n const lines = text.split('\\n');\n const result = [];\n let inTable = false;\n let tableLines = [];\n \n for (let i = 0; i < lines.length; i++) {\n const line = lines[i].trim();\n \n // Check if this line looks like a table row (with or without trailing |)\n if (line.includes('|') && (line.startsWith('|') || /[^\\\\|]/.test(line))) {\n if (!inTable) {\n inTable = true;\n tableLines = [];\n }\n tableLines.push(line);\n } else {\n // Not a table line\n if (inTable) {\n // Process the accumulated table\n const tableHtml = buildTable(tableLines, getAttr);\n if (tableHtml) {\n result.push(tableHtml);\n } else {\n // Not a valid table, restore original lines\n result.push(...tableLines);\n }\n inTable = false;\n tableLines = [];\n }\n result.push(lines[i]);\n }\n }\n \n // Handle table at end of text\n if (inTable && tableLines.length > 0) {\n const tableHtml = buildTable(tableLines, getAttr);\n if (tableHtml) {\n result.push(tableHtml);\n } else {\n result.push(...tableLines);\n }\n }\n \n return result.join('\\n');\n}\n\n/**\n * Build an HTML table from markdown table lines\n */\nfunction buildTable(lines, getAttr) {\n \n if (lines.length < 2) return null;\n \n // Check for separator line (second line should be the separator)\n let separatorIndex = -1;\n for (let i = 1; i < lines.length; i++) {\n // Support separator with or without leading/trailing pipes\n if (/^\\|?[\\s\\-:|]+\\|?$/.test(lines[i]) && lines[i].includes('-')) {\n separatorIndex = i;\n break;\n }\n }\n \n if (separatorIndex === -1) return null;\n \n const headerLines = lines.slice(0, separatorIndex);\n const bodyLines = lines.slice(separatorIndex + 1);\n \n // Parse alignment from separator\n const separator = lines[separatorIndex];\n // Handle pipes at start/end or not\n const separatorCells = separator.trim().replace(/^\\|/, '').replace(/\\|$/, '').split('|');\n const alignments = separatorCells.map(cell => {\n const trimmed = cell.trim();\n if (trimmed.startsWith(':') && trimmed.endsWith(':')) return 'center';\n if (trimmed.endsWith(':')) return 'right';\n return 'left';\n });\n \n let html = `<table${getAttr('table')}>\\n`;\n \n // Build header\n // Note: headerLines will always have length > 0 since separatorIndex starts from 1\n html += `<thead${getAttr('thead')}>\\n`;\n headerLines.forEach(line => {\n html += `<tr${getAttr('tr')}>\\n`;\n // Handle pipes at start/end or not\n const cells = line.trim().replace(/^\\|/, '').replace(/\\|$/, '').split('|');\n cells.forEach((cell, i) => {\n const alignStyle = alignments[i] && alignments[i] !== 'left' ? `text-align:${alignments[i]}` : '';\n const processedCell = processInlineMarkdown(cell.trim(), getAttr);\n html += `<th${getAttr('th', alignStyle)}>${processedCell}</th>\\n`;\n });\n html += '</tr>\\n';\n });\n html += '</thead>\\n';\n \n // Build body\n if (bodyLines.length > 0) {\n html += `<tbody${getAttr('tbody')}>\\n`;\n bodyLines.forEach(line => {\n html += `<tr${getAttr('tr')}>\\n`;\n // Handle pipes at start/end or not\n const cells = line.trim().replace(/^\\|/, '').replace(/\\|$/, '').split('|');\n cells.forEach((cell, i) => {\n const alignStyle = alignments[i] && alignments[i] !== 'left' ? `text-align:${alignments[i]}` : '';\n const processedCell = processInlineMarkdown(cell.trim(), getAttr);\n html += `<td${getAttr('td', alignStyle)}>${processedCell}</td>\\n`;\n });\n html += '</tr>\\n';\n });\n html += '</tbody>\\n';\n }\n \n html += '</table>';\n return html;\n}\n\n/**\n * Process markdown lists (ordered and unordered)\n */\nfunction processLists(text, getAttr, inline_styles, bidirectional) {\n \n const lines = text.split('\\n');\n const result = [];\n const listStack = []; // Track nested lists\n \n // Helper to escape HTML for data-qd attributes. List markers (`-`, `*`,\n // `+`, `1.`, etc.) never contain HTML-special chars, so the replace\n // callback is defensive-only and never actually fires in practice.\n const escapeHtml = (text) => text.replace(/[&<>\"']/g,\n /* istanbul ignore next - defensive: list markers never contain HTML specials */\n m => ({'&':'&','<':'<','>':'>','\"':'"',\"'\":'''})[m]);\n /* istanbul ignore next - trivial no-op fallback; not exercised via bd bundle */\n const dataQd = bidirectional ? (marker) => ` data-qd=\"${escapeHtml(marker)}\"` : () => '';\n \n for (let i = 0; i < lines.length; i++) {\n const line = lines[i];\n const match = line.match(/^(\\s*)([*\\-+]|\\d+\\.)\\s+(.+)$/);\n \n if (match) {\n const [, indent, marker, content] = match;\n const level = Math.floor(indent.length / 2);\n const isOrdered = /^\\d+\\./.test(marker);\n const listType = isOrdered ? 'ol' : 'ul';\n \n // Check for task list items\n let listItemContent = content;\n let taskListClass = '';\n const taskMatch = content.match(/^\\[([x ])\\]\\s+(.*)$/i);\n if (taskMatch && !isOrdered) {\n const [, checked, taskContent] = taskMatch;\n const isChecked = checked.toLowerCase() === 'x';\n const checkboxAttr = inline_styles \n ? ' style=\"margin-right:.5em\"' \n : ` class=\"${CLASS_PREFIX}task-checkbox\"`;\n listItemContent = `<input type=\"checkbox\"${checkboxAttr}${isChecked ? ' checked' : ''} disabled> ${taskContent}`;\n taskListClass = inline_styles ? ' style=\"list-style:none\"' : ` class=\"${CLASS_PREFIX}task-item\"`;\n }\n \n // Close deeper levels\n while (listStack.length > level + 1) {\n const list = listStack.pop();\n result.push(`</${list.type}>`);\n }\n \n // Open new level if needed\n if (listStack.length === level) {\n // Need to open a new list\n listStack.push({ type: listType, level });\n result.push(`<${listType}${getAttr(listType)}>`);\n } else if (listStack.length === level + 1) {\n // Check if we need to switch list type\n const currentList = listStack[listStack.length - 1];\n if (currentList.type !== listType) {\n result.push(`</${currentList.type}>`);\n listStack.pop();\n listStack.push({ type: listType, level });\n result.push(`<${listType}${getAttr(listType)}>`);\n }\n }\n \n const liAttr = taskListClass || getAttr('li');\n result.push(`<li${liAttr}${dataQd(marker)}>${listItemContent}</li>`);\n } else {\n // Not a list item, close all lists\n while (listStack.length > 0) {\n const list = listStack.pop();\n result.push(`</${list.type}>`);\n }\n result.push(line);\n }\n }\n \n // Close any remaining lists\n while (listStack.length > 0) {\n const list = listStack.pop();\n result.push(`</${list.type}>`);\n }\n \n return result.join('\\n');\n}\n\n/**\n * Emit CSS styles for quikdown elements\n * @param {string} prefix - Optional class prefix (default: 'quikdown-')\n * @param {string} theme - Optional theme: 'light' (default) or 'dark'\n * @returns {string} CSS string with quikdown styles\n */\nquikdown.emitStyles = function(prefix = 'quikdown-', theme = 'light') {\n const styles = QUIKDOWN_STYLES;\n \n // Define theme color overrides\n const themeOverrides = {\n dark: {\n '#f4f4f4': '#2a2a2a', // pre background\n '#f0f0f0': '#2a2a2a', // code background\n '#f2f2f2': '#2a2a2a', // th background\n '#ddd': '#3a3a3a', // borders\n '#06c': '#6db3f2', // links\n _textColor: '#e0e0e0'\n },\n light: {\n _textColor: '#333' // Explicit text color for light theme\n }\n };\n \n let css = '';\n for (const [tag, style] of Object.entries(styles)) {\n let themedStyle = style;\n \n // Apply theme overrides if dark theme\n if (theme === 'dark' && themeOverrides.dark) {\n // Replace colors\n for (const [oldColor, newColor] of Object.entries(themeOverrides.dark)) {\n if (!oldColor.startsWith('_')) {\n themedStyle = themedStyle.replace(new RegExp(oldColor, 'g'), newColor);\n }\n }\n \n // Add text color for certain elements in dark theme\n const needsTextColor = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'td', 'li', 'blockquote'];\n if (needsTextColor.includes(tag)) {\n themedStyle += `;color:${themeOverrides.dark._textColor}`;\n }\n } else if (theme === 'light' && themeOverrides.light) {\n // Add explicit text color for light theme elements too\n const needsTextColor = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'td', 'li', 'blockquote'];\n if (needsTextColor.includes(tag)) {\n themedStyle += `;color:${themeOverrides.light._textColor}`;\n }\n }\n \n css += `.${prefix}${tag} { ${themedStyle} }\\n`;\n }\n \n return css;\n};\n\n/**\n * Configure quikdown with options and return a function\n * @param {Object} options - Configuration options\n * @returns {Function} Configured quikdown function\n */\nquikdown.configure = function(options) {\n return function(markdown) {\n return quikdown(markdown, options);\n };\n};\n\n/**\n * Version information\n */\nquikdown.version = quikdownVersion;\n\n// Export for both CommonJS and ES6\n/* istanbul ignore next */\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = quikdown;\n}\n\n// For browser global\n/* istanbul ignore next */\nif (typeof window !== 'undefined') {\n window.quikdown = quikdown;\n}\n\nexport default quikdown;","/**\n * quikdown_bd - Bidirectional markdown/HTML converter\n * Extends core quikdown with HTML→Markdown conversion\n * \n * Uses data-qd attributes to preserve original markdown syntax\n * Enables HTML→Markdown conversion for quikdown-generated HTML\n */\n\nimport quikdown from './quikdown.js';\n\n/**\n * Create bidirectional version by extending quikdown\n * This wraps quikdown and adds the toMarkdown method\n */\nfunction quikdown_bd(markdown, options = {}) {\n // Use core quikdown with bidirectional flag to add data-qd attributes\n return quikdown(markdown, { ...options, bidirectional: true });\n}\n\n// Copy all properties and methods from quikdown (including version).\n// Skip `configure` — quikdown_bd provides its own override below, so the\n// inner quikdown.configure is dead code in this bundle.\nObject.keys(quikdown).forEach(key => {\n if (key === 'configure') return;\n quikdown_bd[key] = quikdown[key];\n});\n\n// Add the toMarkdown method for HTML→Markdown conversion\nquikdown_bd.toMarkdown = function(htmlOrElement, options = {}) {\n // Accept either HTML string or DOM element\n let container;\n if (typeof htmlOrElement === 'string') {\n container = document.createElement('div');\n container.innerHTML = htmlOrElement;\n } else if (htmlOrElement instanceof Element) {\n /* istanbul ignore next - browser-only code path, not testable in jsdom */\n container = htmlOrElement;\n } else {\n return '';\n }\n \n // Walk the DOM tree and reconstruct markdown\n function walkNode(node, parentContext = {}) {\n if (node.nodeType === Node.TEXT_NODE) {\n // Return text content, preserving whitespace where needed\n return node.textContent;\n }\n \n if (node.nodeType !== Node.ELEMENT_NODE) {\n return '';\n }\n \n const tag = node.tagName.toLowerCase();\n const dataQd = node.getAttribute('data-qd');\n \n // Process children with context\n let childContent = '';\n for (const child of node.childNodes) {\n childContent += walkNode(child, { parentTag: tag, ...parentContext });\n }\n \n // Determine markdown based on element and attributes\n switch (tag) {\n case 'h1':\n case 'h2':\n case 'h3':\n case 'h4':\n case 'h5':\n case 'h6':\n const level = parseInt(tag[1]);\n const prefix = dataQd || '#'.repeat(level);\n return `${prefix} ${childContent.trim()}\\n\\n`;\n \n case 'strong':\n case 'b':\n if (!childContent) return ''; // Don't add markers for empty content\n const boldMarker = dataQd || '**';\n return `${boldMarker}${childContent}${boldMarker}`;\n \n case 'em':\n case 'i':\n if (!childContent) return ''; // Don't add markers for empty content\n const emMarker = dataQd || '*';\n return `${emMarker}${childContent}${emMarker}`;\n \n case 'del':\n case 's':\n case 'strike':\n if (!childContent) return ''; // Don't add markers for empty content\n const delMarker = dataQd || '~~';\n return `${delMarker}${childContent}${delMarker}`;\n \n case 'code':\n // Note: code inside pre is handled directly by the pre case using querySelector\n if (!childContent) return ''; // Don't add markers for empty content\n const codeMarker = dataQd || '`';\n return `${codeMarker}${childContent}${codeMarker}`;\n \n case 'pre':\n const fence = node.getAttribute('data-qd-fence') || dataQd || '```';\n const lang = node.getAttribute('data-qd-lang') || '';\n \n // Check if this was created by a fence plugin with reverse handler\n if (options.fence_plugin && options.fence_plugin.reverse && lang) {\n try {\n const result = options.fence_plugin.reverse(node);\n if (result && result.content) {\n const fenceMarker = result.fence || fence;\n const langStr = result.lang || lang;\n return `${fenceMarker}${langStr}\\n${result.content}\\n${fenceMarker}\\n\\n`;\n }\n } catch (err) {\n console.warn('Fence reverse handler error:', err);\n // Fall through to default handling\n }\n }\n \n // Fallback: use data-qd-source if available\n const source = node.getAttribute('data-qd-source');\n if (source) {\n return `${fence}${lang}\\n${source}\\n${fence}\\n\\n`;\n }\n \n // Final fallback: extract text content\n const codeEl = node.querySelector('code');\n const codeContent = codeEl ? codeEl.textContent : childContent;\n return `${fence}${lang}\\n${codeContent.trimEnd()}\\n${fence}\\n\\n`;\n \n case 'blockquote':\n const quoteMarker = dataQd || '>';\n const lines = childContent.trim().split('\\n');\n return lines.map(line => `${quoteMarker} ${line}`).join('\\n') + '\\n\\n';\n \n case 'hr':\n const hrMarker = dataQd || '---';\n return `${hrMarker}\\n\\n`;\n \n case 'br':\n const brMarker = dataQd || ' ';\n return `${brMarker}\\n`;\n \n case 'a':\n const linkText = node.getAttribute('data-qd-text') || childContent.trim();\n const href = node.getAttribute('href') || '';\n // Check for autolinks\n if (linkText === href && !dataQd) {\n return `<${href}>`;\n }\n return `[${linkText}](${href})`;\n \n case 'img':\n const alt = node.getAttribute('data-qd-alt') || node.getAttribute('alt') || '';\n const src = node.getAttribute('data-qd-src') || node.getAttribute('src') || '';\n const imgMarker = dataQd || '!';\n return `${imgMarker}[${alt}](${src})`;\n \n case 'ul':\n case 'ol':\n return walkList(node, tag === 'ol') + '\\n';\n \n case 'li':\n // Handled by list processor\n return childContent;\n \n case 'table':\n return walkTable(node) + '\\n\\n';\n \n case 'p':\n // Check if it's actually a paragraph or just a wrapper\n if (childContent.trim()) {\n // Check if paragraph ends with a line that's just whitespace\n // This indicates an intentional blank line before the next element\n const lines = childContent.split('\\n');\n let content = childContent.trim();\n \n // If the last line(s) are just whitespace, preserve one blank line\n if (lines.length > 1) {\n let trailingBlankLines = 0;\n for (let i = lines.length - 1; i >= 0; i--) {\n if (lines[i].trim() === '') {\n trailingBlankLines++;\n } else {\n break;\n }\n }\n if (trailingBlankLines > 0) {\n // Add a line with just a space, followed by single newline\n // The \\n\\n will be added below for paragraph separation\n content = content + '\\n ';\n // Only add one newline since we're preserving the space line\n return content + '\\n';\n }\n }\n \n return content + '\\n\\n';\n }\n return '';\n \n case 'div':\n // Check if this was created by a fence plugin with reverse handler\n const divLang = node.getAttribute('data-qd-lang');\n const divFence = node.getAttribute('data-qd-fence');\n \n if (divLang && options.fence_plugin && options.fence_plugin.reverse) {\n try {\n const result = options.fence_plugin.reverse(node);\n if (result && result.content) {\n const fenceMarker = result.fence || divFence || '```';\n const langStr = result.lang || divLang;\n return `${fenceMarker}${langStr}\\n${result.content}\\n${fenceMarker}\\n\\n`;\n }\n } catch (err) {\n console.warn('Fence reverse handler error:', err);\n // Fall through to default handling\n }\n }\n \n // Fallback: use data-qd-source if available\n const divSource = node.getAttribute('data-qd-source');\n if (divSource && divFence) {\n return `${divFence}${divLang || ''}\\n${divSource}\\n${divFence}\\n\\n`;\n }\n \n // Check if it's a mermaid container\n if (node.classList && node.classList.contains('mermaid-container')) {\n const fence = node.getAttribute('data-qd-fence') || '```';\n const lang = node.getAttribute('data-qd-lang') || 'mermaid';\n \n // First check for data-qd-source attribute on the container\n const source = node.getAttribute('data-qd-source');\n if (source) {\n // Decode HTML entities from the attribute (mainly ")\n const temp = document.createElement('textarea');\n temp.innerHTML = source;\n const code = temp.value;\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n \n // Check for source on the pre.mermaid element\n const mermaidPre = node.querySelector('pre.mermaid');\n if (mermaidPre) {\n const preSource = mermaidPre.getAttribute('data-qd-source');\n if (preSource) {\n const temp = document.createElement('textarea');\n temp.innerHTML = preSource;\n const code = temp.value;\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n }\n \n // Fallback: Look for the legacy .mermaid-source element\n const sourceElement = node.querySelector('.mermaid-source');\n if (sourceElement) {\n // Decode HTML entities\n const temp = document.createElement('div');\n temp.innerHTML = sourceElement.innerHTML;\n const code = temp.textContent;\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n \n // Final fallback: try to extract from the mermaid element (unreliable after rendering)\n const mermaidElement = node.querySelector('.mermaid');\n if (mermaidElement && mermaidElement.textContent.includes('graph')) {\n return `${fence}${lang}\\n${mermaidElement.textContent.trim()}\\n${fence}\\n\\n`;\n }\n }\n // Check if it's a standalone mermaid diagram (legacy)\n if (node.classList && node.classList.contains('mermaid')) {\n const fence = node.getAttribute('data-qd-fence') || '```';\n const lang = node.getAttribute('data-qd-lang') || 'mermaid';\n const code = node.textContent.trim();\n return `${fence}${lang}\\n${code}\\n${fence}\\n\\n`;\n }\n // Pass through other divs\n return childContent;\n \n case 'span':\n // Pass through container elements\n return childContent;\n \n default:\n return childContent;\n }\n }\n \n // Walk list elements\n function walkList(listNode, isOrdered, depth = 0) {\n let result = '';\n let index = 1;\n const indent = ' '.repeat(depth);\n \n for (const child of listNode.children) {\n if (child.tagName !== 'LI') continue;\n \n const dataQd = child.getAttribute('data-qd');\n let marker = dataQd || (isOrdered ? `${index}.` : '-');\n \n // Check for task list checkbox\n const checkbox = child.querySelector('input[type=\"checkbox\"]');\n if (checkbox) {\n const checked = checkbox.checked ? 'x' : ' ';\n marker = '-';\n // Get text without the checkbox\n let text = '';\n for (const node of child.childNodes) {\n if (node.nodeType === Node.TEXT_NODE) {\n text += node.textContent;\n } else if (node.tagName && node.tagName !== 'INPUT') {\n text += walkNode(node);\n }\n }\n result += `${indent}${marker} [${checked}] ${text.trim()}\\n`;\n } else {\n let itemContent = '';\n \n for (const node of child.childNodes) {\n if (node.tagName === 'UL' || node.tagName === 'OL') {\n itemContent += walkList(node, node.tagName === 'OL', depth + 1);\n } else {\n itemContent += walkNode(node);\n }\n }\n \n result += `${indent}${marker} ${itemContent.trim()}\\n`;\n }\n \n index++;\n }\n \n return result;\n }\n \n // Walk table elements\n function walkTable(table) {\n let result = '';\n const alignData = table.getAttribute('data-qd-align');\n const alignments = alignData ? alignData.split(',') : [];\n \n // Process header\n const thead = table.querySelector('thead');\n if (thead) {\n const headerRow = thead.querySelector('tr');\n if (headerRow) {\n const headers = [];\n for (const th of headerRow.querySelectorAll('th')) {\n headers.push(th.textContent.trim());\n }\n result += '| ' + headers.join(' | ') + ' |\\n';\n \n // Add separator with alignment\n const separators = headers.map((_, i) => {\n const align = alignments[i] || 'left';\n if (align === 'center') return ':---:';\n if (align === 'right') return '---:';\n return '---';\n });\n result += '| ' + separators.join(' | ') + ' |\\n';\n }\n }\n \n // Process body\n const tbody = table.querySelector('tbody');\n if (tbody) {\n for (const row of tbody.querySelectorAll('tr')) {\n const cells = [];\n for (const td of row.querySelectorAll('td')) {\n cells.push(td.textContent.trim());\n }\n if (cells.length > 0) {\n result += '| ' + cells.join(' | ') + ' |\\n';\n }\n }\n }\n \n return result.trim();\n }\n \n // Process the DOM tree\n let markdown = walkNode(container);\n \n // Clean up\n markdown = markdown.replace(/\\n{3,}/g, '\\n\\n'); // Remove excessive newlines\n markdown = markdown.trim();\n \n return markdown;\n};\n\n// Override the configure method to return a bidirectional version.\n// We delegate to the inner quikdown.configure so the shared closure\n// machinery is exercised in both bundles (no dead code).\nquikdown_bd.configure = function(options) {\n const innerParser = quikdown.configure({ ...options, bidirectional: true });\n return function(markdown) {\n return innerParser(markdown);\n };\n};\n\n// Set version\n// Version is already copied from quikdown via Object.keys loop\n\n// Export for both module and browser\n/* istanbul ignore next */\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = quikdown_bd;\n}\n\n/* istanbul ignore next */\nif (typeof window !== 'undefined') {\n window.quikdown_bd = quikdown_bd;\n}\n\nexport default quikdown_bd;","/**\n * Rich copy functionality for QuikdownEditor\n * Handles copying rendered content with proper formatting for pasting into rich text editors\n */\n\n/**\n * Get platform information\n * @returns {string} The detected platform: 'macos', 'windows', 'linux', or 'unknown'\n */\nfunction getPlatform() {\n const platform = navigator.platform?.toLowerCase() || '';\n const userAgent = navigator.userAgent?.toLowerCase() || '';\n \n if (platform.includes('mac') || userAgent.includes('mac')) {\n return 'macos';\n } else if (userAgent.includes('windows')) {\n return 'windows';\n } else if (userAgent.includes('linux')) {\n return 'linux';\n }\n return 'unknown';\n}\n\n/**\n * Copy to clipboard using HTML selection fallback (for Safari)\n * Uses div with selection to preserve HTML formatting\n * @param {string} html - HTML content to copy\n * @returns {boolean} Success status\n */\nfunction copyToClipboard(html) {\n let tempDiv;\n let result;\n \n try {\n // Use a div instead of textarea to preserve HTML formatting\n tempDiv = document.createElement('div');\n tempDiv.style.position = 'fixed';\n tempDiv.style.left = '-9999px';\n tempDiv.style.top = '0';\n tempDiv.style.width = '1px';\n tempDiv.style.height = '1px';\n tempDiv.style.overflow = 'hidden';\n tempDiv.innerHTML = html;\n \n document.body.appendChild(tempDiv);\n \n // Select the HTML content\n const range = document.createRange();\n range.selectNodeContents(tempDiv);\n const selection = window.getSelection();\n selection.removeAllRanges();\n selection.addRange(range);\n \n // Try to copy\n result = document.execCommand('copy');\n \n // Clear selection\n selection.removeAllRanges();\n } catch (err) {\n console.error('Fallback copy failed:', err);\n result = false;\n } finally {\n if (tempDiv && tempDiv.parentNode) {\n document.body.removeChild(tempDiv);\n }\n }\n \n return result;\n}\n\n/**\n * Convert SVG to PNG blob (based on squibview's implementation)\n * @param {SVGElement} svgElement - The SVG element to convert\n * @returns {Promise<Blob>} A promise that resolves with the PNG blob\n */\nasync function svgToPng(svgElement, needsWhiteBackground = false) {\n return new Promise((resolve, reject) => {\n const svgString = new XMLSerializer().serializeToString(svgElement);\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n const img = new Image();\n \n const scale = 2;\n \n // Check if this is a Mermaid-generated SVG (they don't have explicit width/height attributes)\n const isMermaidSvg = svgElement.closest('.mermaid') || svgElement.classList.contains('mermaid');\n const hasExplicitDimensions = svgElement.getAttribute('width') && svgElement.getAttribute('height');\n \n let svgWidth, svgHeight;\n \n if (isMermaidSvg || !hasExplicitDimensions) {\n // For Mermaid or other generated SVGs, prioritize computed dimensions\n svgWidth = svgElement.clientWidth || \n (svgElement.viewBox && svgElement.viewBox.baseVal.width) || \n parseFloat(svgElement.getAttribute('width')) || 400;\n svgHeight = svgElement.clientHeight || \n (svgElement.viewBox && svgElement.viewBox.baseVal.height) || \n parseFloat(svgElement.getAttribute('height')) || 300;\n } else {\n // For explicit SVGs (like fenced SVG blocks), prioritize explicit attributes\n svgWidth = parseFloat(svgElement.getAttribute('width')) || \n (svgElement.viewBox && svgElement.viewBox.baseVal.width) || \n svgElement.clientWidth || 400;\n svgHeight = parseFloat(svgElement.getAttribute('height')) || \n (svgElement.viewBox && svgElement.viewBox.baseVal.height) || \n svgElement.clientHeight || 300;\n }\n \n // Ensure the SVG string has explicit dimensions by modifying it if necessary\n let modifiedSvgString = svgString;\n if (svgWidth && svgHeight) {\n // Create a temporary SVG element to modify the serialized string\n const tempDiv = document.createElement('div');\n tempDiv.innerHTML = svgString;\n const tempSvg = tempDiv.querySelector('svg');\n if (tempSvg) {\n tempSvg.setAttribute('width', svgWidth.toString());\n tempSvg.setAttribute('height', svgHeight.toString());\n modifiedSvgString = new XMLSerializer().serializeToString(tempSvg);\n }\n }\n \n canvas.width = svgWidth * scale;\n canvas.height = svgHeight * scale;\n ctx.scale(scale, scale);\n \n img.onload = () => {\n try {\n // Add white background for math equations (they often have transparent backgrounds)\n if (needsWhiteBackground) {\n ctx.fillStyle = 'white';\n ctx.fillRect(0, 0, canvas.width, canvas.height);\n }\n \n ctx.drawImage(img, 0, 0, svgWidth, svgHeight);\n canvas.toBlob(blob => {\n resolve(blob);\n }, 'image/png', 1.0);\n } catch (err) {\n reject(err);\n }\n };\n \n img.onerror = reject;\n // Use data URI instead of blob URL to avoid tainting the canvas\n const svgDataUrl = `data:image/svg+xml;charset=utf-8,${encodeURIComponent(modifiedSvgString)}`;\n img.src = svgDataUrl;\n });\n}\n\n/**\n * Rasterize a GeoJSON Leaflet map to PNG data URL (following Gem's guide)\n * @param {HTMLElement} liveContainer - The live map container element\n * @returns {Promise<string|null>} PNG data URL or null if failed\n */\nasync function rasterizeGeoJSONMap(liveContainer) {\n try {\n const map = liveContainer._map;\n if (!map) {\n console.warn('No map found on container');\n return null;\n }\n \n // Get container dimensions\n const mapRect = liveContainer.getBoundingClientRect();\n const width = Math.round(mapRect.width);\n const height = Math.round(mapRect.height);\n \n if (width === 0 || height === 0) {\n console.warn('Map container has zero dimensions');\n return null;\n }\n \n // Create canvas sized to the map container\n const canvas = document.createElement('canvas');\n const dpr = window.devicePixelRatio || 1;\n \n // Set canvas size with DPR for sharpness\n canvas.width = width * dpr;\n canvas.height = height * dpr;\n canvas.style.width = width + 'px';\n canvas.style.height = height + 'px';\n \n const ctx = canvas.getContext('2d');\n ctx.scale(dpr, dpr);\n \n // White background\n ctx.fillStyle = '#FFFFFF';\n ctx.fillRect(0, 0, width, height);\n \n // 1. Draw tiles from THIS container only\n const tiles = liveContainer.querySelectorAll('.leaflet-tile');\n \n const tilePromises = [];\n for (const tile of tiles) {\n tilePromises.push(new Promise((resolve) => {\n const img = new Image();\n img.crossOrigin = 'anonymous';\n \n img.onload = () => {\n try {\n // Calculate tile position relative to container\n const tileRect = tile.getBoundingClientRect();\n const offsetX = tileRect.left - mapRect.left;\n const offsetY = tileRect.top - mapRect.top;\n \n // Draw tile at correct position\n ctx.drawImage(img, offsetX, offsetY, tileRect.width, tileRect.height);\n } catch (err) {\n console.warn('Failed to draw tile:', err);\n }\n resolve();\n };\n \n img.onerror = () => {\n console.warn('Failed to load tile:', tile.src);\n resolve();\n };\n \n img.src = tile.src;\n }));\n }\n \n // Wait for all tiles to load\n await Promise.all(tilePromises);\n \n // 2. Draw vector overlays (SVG paths for GeoJSON features)\n const svgOverlays = liveContainer.querySelectorAll('svg:not(.leaflet-attribution-flag)');\n \n for (const svg of svgOverlays) {\n // Skip attribution/control overlays\n if (svg.closest('.leaflet-control')) continue;\n \n try {\n const svgRect = svg.getBoundingClientRect();\n const offsetX = svgRect.left - mapRect.left;\n const offsetY = svgRect.top - mapRect.top;\n \n // Serialize SVG\n const serializer = new XMLSerializer();\n const svgStr = serializer.serializeToString(svg);\n const svgBlob = new Blob([svgStr], { type: 'image/svg+xml;charset=utf-8' });\n const url = URL.createObjectURL(svgBlob);\n \n // Draw SVG overlay\n await new Promise((resolve, reject) => {\n const img = new Image();\n img.onload = () => {\n ctx.drawImage(img, offsetX, offsetY, svgRect.width, svgRect.height);\n URL.revokeObjectURL(url);\n resolve();\n };\n img.onerror = () => {\n URL.revokeObjectURL(url);\n reject(new Error('Failed to load SVG overlay'));\n };\n img.src = url;\n });\n } catch (err) {\n console.warn('Failed to draw SVG overlay:', err);\n }\n }\n \n // 3. Draw marker icons if any\n const markerIcons = liveContainer.querySelectorAll('.leaflet-marker-icon');\n \n for (const marker of markerIcons) {\n try {\n const img = new Image();\n img.crossOrigin = 'anonymous';\n \n await new Promise((resolve) => {\n img.onload = () => {\n const markerRect = marker.getBoundingClientRect();\n const offsetX = markerRect.left - mapRect.left;\n const offsetY = markerRect.top - mapRect.top;\n ctx.drawImage(img, offsetX, offsetY, markerRect.width, markerRect.height);\n resolve();\n };\n img.onerror = resolve;\n img.src = marker.src;\n });\n } catch (err) {\n console.warn('Failed to draw marker icon:', err);\n }\n }\n \n // Return PNG data URL\n return canvas.toDataURL('image/png', 1.0);\n \n } catch (error) {\n console.error('Failed to rasterize GeoJSON map:', error);\n return null;\n }\n}\n\n/**\n * Get rendered content as rich HTML suitable for clipboard\n * @param {HTMLElement} previewPanel - The preview panel element to copy from\n * @returns {Promise<{success: boolean, html?: string, text?: string}>}\n */\nexport async function getRenderedContent(previewPanel) {\n if (!previewPanel) {\n throw new Error('No preview panel available');\n }\n \n // Check if MathJax needs to render (only if not already rendered)\n const mathBlocks = previewPanel.querySelectorAll('.math-display');\n if (mathBlocks.length > 0) {\n // Check if already rendered (has mjx-container inside)\n const needsRendering = Array.from(mathBlocks).some(block => !block.querySelector('mjx-container'));\n \n if (needsRendering && window.MathJax && window.MathJax.typesetPromise) {\n try {\n await window.MathJax.typesetPromise(Array.from(mathBlocks));\n } catch (err) {\n console.warn('MathJax typesetting failed:', err);\n }\n }\n }\n \n // Clone the preview panel to avoid modifying the actual DOM\n const clone = previewPanel.cloneNode(true);\n \n // Process different fence types for rich copy\n try {\n // Phase 1: Process basic markdown elements with inline styles\n \n // 1.1 Text formatting - add inline styles\n clone.querySelectorAll('strong, b').forEach(el => {\n el.style.fontWeight = 'bold';\n });\n \n clone.querySelectorAll('em, i').forEach(el => {\n el.style.fontStyle = 'italic';\n });\n \n clone.querySelectorAll('del, s, strike').forEach(el => {\n el.style.textDecoration = 'line-through';\n });\n \n clone.querySelectorAll('u').forEach(el => {\n el.style.textDecoration = 'underline';\n });\n \n clone.querySelectorAll('code:not(pre code)').forEach(el => {\n el.style.backgroundColor = '#f4f4f4';\n el.style.padding = '2px 4px';\n el.style.borderRadius = '3px';\n el.style.fontFamily = 'monospace';\n el.style.fontSize = '0.9em';\n });\n \n // 1.2 Block elements - add inline styles\n clone.querySelectorAll('h1').forEach(el => {\n el.style.fontSize = '2em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '0.67em';\n el.style.marginBottom = '0.67em';\n });\n \n clone.querySelectorAll('h2').forEach(el => {\n el.style.fontSize = '1.5em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '0.83em';\n el.style.marginBottom = '0.83em';\n });\n \n clone.querySelectorAll('h3').forEach(el => {\n el.style.fontSize = '1.17em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '1em';\n el.style.marginBottom = '1em';\n });\n \n clone.querySelectorAll('h4').forEach(el => {\n el.style.fontSize = '1em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '1.33em';\n el.style.marginBottom = '1.33em';\n });\n \n clone.querySelectorAll('h5').forEach(el => {\n el.style.fontSize = '0.83em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '1.67em';\n el.style.marginBottom = '1.67em';\n });\n \n clone.querySelectorAll('h6').forEach(el => {\n el.style.fontSize = '0.67em';\n el.style.fontWeight = 'bold';\n el.style.marginTop = '2.33em';\n el.style.marginBottom = '2.33em';\n });\n \n clone.querySelectorAll('blockquote').forEach(el => {\n el.style.borderLeft = '4px solid #ddd';\n el.style.marginLeft = '0';\n el.style.paddingLeft = '1em';\n el.style.color = '#666';\n });\n \n clone.querySelectorAll('hr').forEach(el => {\n el.style.border = 'none';\n el.style.borderTop = '1px solid #ccc';\n el.style.margin = '1em 0';\n });\n \n // 1.3 Tables - add inline styles\n clone.querySelectorAll('table').forEach(table => {\n table.style.borderCollapse = 'collapse';\n table.style.width = '100%';\n table.style.marginBottom = '1em';\n });\n \n clone.querySelectorAll('th').forEach(th => {\n th.style.border = '1px solid #ccc';\n th.style.padding = '8px';\n th.style.textAlign = 'left';\n th.style.backgroundColor = '#f0f0f0';\n th.style.fontWeight = 'bold';\n });\n \n clone.querySelectorAll('td').forEach(td => {\n td.style.border = '1px solid #ccc';\n td.style.padding = '8px';\n td.style.textAlign = 'left';\n });\n \n // 1.4 Links - add inline styles\n clone.querySelectorAll('a').forEach(a => {\n a.style.color = '#0066cc';\n a.style.textDecoration = 'underline';\n });\n \n // Process code blocks - wrap in table and add syntax highlighting colors\n clone.querySelectorAll('pre code').forEach(block => {\n const pre = block.parentElement;\n \n // Add inline styles for syntax highlighting (GitHub theme colors)\n if (block.classList.contains('hljs')) {\n // Apply inline styles to all highlight.js elements\n block.querySelectorAll('.hljs-keyword').forEach(el => {\n el.style.color = '#d73a49';\n el.style.fontWeight = 'bold';\n });\n block.querySelectorAll('.hljs-string').forEach(el => {\n el.style.color = '#032f62';\n });\n block.querySelectorAll('.hljs-number').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-comment').forEach(el => {\n el.style.color = '#6a737d';\n el.style.fontStyle = 'italic';\n });\n block.querySelectorAll('.hljs-function').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-class').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-title').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-built_in').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-literal').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-meta').forEach(el => {\n el.style.color = '#005cc5';\n });\n block.querySelectorAll('.hljs-attr').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-variable').forEach(el => {\n el.style.color = '#e36209';\n });\n block.querySelectorAll('.hljs-regexp').forEach(el => {\n el.style.color = '#032f62';\n });\n block.querySelectorAll('.hljs-selector-class').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-selector-id').forEach(el => {\n el.style.color = '#6f42c1';\n });\n block.querySelectorAll('.hljs-selector-tag').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-tag').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-name').forEach(el => {\n el.style.color = '#22863a';\n });\n block.querySelectorAll('.hljs-attribute').forEach(el => {\n el.style.color = '#6f42c1';\n });\n }\n \n const table = document.createElement('table');\n table.style.width = '100%';\n table.style.borderCollapse = 'collapse';\n table.style.border = 'none';\n table.style.marginBottom = '1em';\n \n const tr = document.createElement('tr');\n const td = document.createElement('td');\n td.style.backgroundColor = '#f7f7f7';\n td.style.padding = '12px';\n td.style.fontFamily = 'Consolas, Monaco, \"Courier New\", monospace';\n td.style.fontSize = '14px';\n td.style.lineHeight = '1.4';\n td.style.whiteSpace = 'pre';\n td.style.overflowX = 'auto';\n td.style.border = '1px solid #ddd';\n td.style.borderRadius = '4px';\n \n // Move the formatted code content with inline styles\n td.innerHTML = block.innerHTML;\n \n tr.appendChild(td);\n table.appendChild(tr);\n \n // Replace the pre element with the table\n pre.parentNode.replaceChild(table, pre);\n });\n \n // Process images - convert to data URLs and ensure proper dimensions\n const images = clone.querySelectorAll('img');\n for (const img of images) {\n // Ensure image has dimensions for Google Docs compatibility\n if (!img.width && img.naturalWidth) {\n img.width = img.naturalWidth;\n }\n if (!img.height && img.naturalHeight) {\n img.height = img.naturalHeight;\n }\n \n // Set max dimensions to prevent huge images\n const maxWidth = 800;\n const maxHeight = 600;\n if (img.width > maxWidth || img.height > maxHeight) {\n const scale = Math.min(maxWidth / img.width, maxHeight / img.height);\n img.width = Math.round(img.width * scale);\n img.height = Math.round(img.height * scale);\n }\n \n // Ensure width and height attributes are set\n if (img.width) {\n img.setAttribute('width', img.width.toString());\n img.style.width = img.width + 'px';\n }\n if (img.height) {\n img.setAttribute('height', img.height.toString());\n img.style.height = img.height + 'px';\n }\n \n // Add v:shapes for Word compatibility\n if (!img.getAttribute('v:shapes')) {\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n }\n \n // Skip if already a data URL\n if (img.src && !img.src.startsWith('data:')) {\n try {\n // Try to convert to data URL\n const response = await fetch(img.src);\n const blob = await response.blob();\n \n // Check if image is too large (Google Docs has limits)\n const maxSize = 2 * 1024 * 1024; // 2MB limit for inline images\n if (blob.size > maxSize) {\n console.warn('Image too large for inline data URL:', img.src, 'Size:', blob.size);\n // For large images, we might want to resize or keep the URL\n continue;\n }\n \n const dataUrl = await new Promise(resolve => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result);\n reader.readAsDataURL(blob);\n });\n img.src = dataUrl;\n } catch (err) {\n console.warn('Failed to convert image to data URL:', img.src, err);\n // Keep original src if conversion fails\n }\n }\n }\n \n // Phase 2: Process fence block types\n // 1. Process STL 3D models - convert canvas to image or placeholder\n const stlContainers = clone.querySelectorAll('.qde-stl-container');\n for (const container of stlContainers) {\n try {\n // Find the corresponding original container to get the canvas\n const containerId = container.dataset.stlId;\n const originalContainer = previewPanel.querySelector(`.qde-stl-container[data-stl-id=\"${containerId}\"]`);\n \n if (originalContainer) {\n // Look for canvas element in the original container (Three.js WebGL canvas)\n const canvas = originalContainer.querySelector('canvas');\n if (canvas && canvas.width > 0 && canvas.height > 0) {\n try {\n // Get Three.js references stored on the container (like squibview)\n const renderer = originalContainer._threeRenderer;\n const scene = originalContainer._threeScene;\n const camera = originalContainer._threeCamera;\n \n // If we have access to the Three.js objects, force render the scene\n if (renderer && scene && camera) {\n renderer.render(scene, camera);\n }\n \n // Try to capture the canvas as an image\n const dataUrl = canvas.toDataURL('image/png', 1.0);\n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Use canvas dimensions for the image\n const imgWidth = canvas.width / 2; // Divide by scale factor (2x for retina)\n const imgHeight = canvas.height / 2;\n \n // Set both HTML attributes and CSS properties for maximum compatibility\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none';\n img.style.maxHeight = 'none';\n img.style.border = '1px solid #ddd';\n img.style.borderRadius = '4px';\n img.style.margin = '0.5em 0';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'STL 3D Model';\n \n container.parentNode.replaceChild(img, container);\n continue;\n } catch (canvasErr) {\n console.warn('Failed to convert STL canvas to image (likely WebGL context issue):', canvasErr);\n }\n } else {\n console.warn('No valid canvas found in STL container');\n }\n } else {\n console.warn('Could not find original STL container');\n }\n } catch (err) {\n console.error('Error processing STL container for copy:', err);\n }\n \n // Fallback to placeholder if canvas conversion fails\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[STL 3D Model - Interactive content not available in copy]';\n container.parentNode.replaceChild(placeholder, container);\n }\n \n // 2. Process Mermaid diagrams - convert SVG to PNG\n const mermaidContainers = clone.querySelectorAll('.mermaid');\n for (const container of mermaidContainers) {\n const svg = container.querySelector('svg');\n if (svg) {\n try {\n const pngBlob = await svgToPng(svg);\n const dataUrl = await new Promise(resolve => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result);\n reader.readAsDataURL(pngBlob);\n });\n \n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Use the exact same dimension calculation logic as svgToPng (like squibview)\n const isMermaidSvg = svg.closest('.mermaid') || svg.classList.contains('mermaid');\n const hasExplicitDimensions = svg.getAttribute('width') && svg.getAttribute('height');\n \n let imgWidth, imgHeight;\n \n if (isMermaidSvg || !hasExplicitDimensions) {\n // For Mermaid or other generated SVGs, prioritize computed dimensions\n imgWidth = svg.clientWidth || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n parseFloat(svg.getAttribute('width')) || 400;\n imgHeight = svg.clientHeight || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n parseFloat(svg.getAttribute('height')) || 300;\n } else {\n // For explicit SVGs (like fenced SVG blocks), prioritize explicit attributes\n imgWidth = parseFloat(svg.getAttribute('width')) || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n svg.clientWidth || 400;\n imgHeight = parseFloat(svg.getAttribute('height')) || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n svg.clientHeight || 300;\n }\n \n // Set both HTML attributes and CSS properties for maximum compatibility (like squibview)\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none'; // Prevent CSS from constraining the image\n img.style.maxHeight = 'none';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'Mermaid Diagram';\n \n container.parentNode.replaceChild(img, container);\n } catch (err) {\n console.warn('Failed to convert Mermaid diagram:', err);\n // Fallback: leave as SVG\n }\n }\n }\n \n // 3. Process Chart.js charts - convert canvas to image\n const chartContainers = clone.querySelectorAll('.qde-chart-container');\n for (const container of chartContainers) {\n try {\n const containerId = container.dataset.chartId;\n const originalContainer = previewPanel.querySelector(`.qde-chart-container[data-chart-id=\"${containerId}\"]`);\n \n if (originalContainer) {\n const canvas = originalContainer.querySelector('canvas');\n if (canvas && canvas.width > 0 && canvas.height > 0) {\n try {\n const dataUrl = canvas.toDataURL('image/png', 1.0);\n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Use canvas dimensions for the image\n const imgWidth = canvas.width;\n const imgHeight = canvas.height;\n \n // Set both HTML attributes and CSS properties for maximum compatibility\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none';\n img.style.maxHeight = 'none';\n img.style.margin = '0.5em 0';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'Chart';\n \n container.parentNode.replaceChild(img, container);\n continue;\n } catch (canvasErr) {\n console.warn('Failed to convert chart canvas to image:', canvasErr);\n }\n }\n }\n } catch (err) {\n console.warn('Error processing chart container:', err);\n }\n \n // Fallback to placeholder\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[Chart - Interactive content not available in copy]';\n container.parentNode.replaceChild(placeholder, container);\n }\n \n // 4. Process SVG fenced blocks - convert to PNG\n const svgContainers = clone.querySelectorAll('.qde-svg-container svg');\n for (const svg of svgContainers) {\n try {\n const pngBlob = await svgToPng(svg);\n const dataUrl = await new Promise(resolve => {\n const reader = new FileReader();\n reader.onloadend = () => resolve(reader.result);\n reader.readAsDataURL(pngBlob);\n });\n \n const img = document.createElement('img');\n img.src = dataUrl;\n \n // Calculate dimensions for the SVG\n const hasExplicitDimensions = svg.getAttribute('width') && svg.getAttribute('height');\n \n let imgWidth, imgHeight;\n \n if (hasExplicitDimensions) {\n // For explicit SVGs (like fenced SVG blocks), prioritize explicit attributes\n imgWidth = parseFloat(svg.getAttribute('width')) || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n svg.clientWidth || 400;\n imgHeight = parseFloat(svg.getAttribute('height')) || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n svg.clientHeight || 300;\n } else {\n // For generated SVGs, prioritize computed dimensions\n imgWidth = svg.clientWidth || \n (svg.viewBox && svg.viewBox.baseVal.width) || \n parseFloat(svg.getAttribute('width')) || 400;\n imgHeight = svg.clientHeight || \n (svg.viewBox && svg.viewBox.baseVal.height) || \n parseFloat(svg.getAttribute('height')) || 300;\n }\n \n // Set both HTML attributes and CSS properties for maximum compatibility\n img.width = imgWidth;\n img.height = imgHeight;\n img.setAttribute('width', imgWidth.toString());\n img.setAttribute('height', imgHeight.toString());\n img.style.width = imgWidth + 'px';\n img.style.height = imgHeight + 'px';\n img.style.maxWidth = 'none'; // Prevent CSS from constraining the image\n img.style.maxHeight = 'none';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'SVG Image';\n \n svg.parentNode.replaceChild(img, svg);\n } catch (err) {\n console.warn('Failed to convert SVG to image:', err);\n // Leave as SVG if conversion fails\n }\n }\n \n // 5. Process Math equations - convert to PNG images (exactly like SquibView)\n const mathElements = Array.from(clone.querySelectorAll('.math-display'));\n \n if (mathElements.length > 0) {\n for (const mathEl of mathElements) {\n try {\n // Find SVG inside the math element (MathJax creates it)\n const svg = mathEl.querySelector('svg');\n if (!svg) {\n console.warn('No SVG found in math element, skipping');\n continue;\n }\n \n // Convert SVG to PNG data URL (exactly like SquibView)\n const serializer = new XMLSerializer();\n const svgStr = serializer.serializeToString(svg);\n const svgBlob = new Blob([svgStr], { type: \"image/svg+xml;charset=utf-8\" });\n const url = URL.createObjectURL(svgBlob);\n \n const img = new Image();\n const dataUrl = await new Promise((resolve, reject) => {\n img.onload = function () {\n const canvas = document.createElement('canvas');\n \n // Determine SVG dimensions robustly (exactly like SquibView)\n let width, height;\n try {\n // First try baseVal.value (works for absolute units)\n width = svg.width.baseVal.value;\n height = svg.height.baseVal.value;\n } catch (_e) {\n // Fallback for relative units - use viewBox or rendered size\n if (svg.viewBox && svg.viewBox.baseVal) {\n width = svg.viewBox.baseVal.width;\n height = svg.viewBox.baseVal.height;\n } else {\n // Use the natural size of the loaded image\n width = img.naturalWidth || img.width || 200;\n height = img.naturalHeight || img.height || 50;\n }\n }\n \n // Scale down for much smaller paste sizes\n const targetMaxWidth = 150; // Further reduced\n const targetMaxHeight = 45; // Further reduced\n \n // Apply aggressive downsizing for MathJax SVGs\n let scaleFactor = 0.04; // Further reduced for smaller output\n \n const scaledWidth = width * scaleFactor;\n const scaledHeight = height * scaleFactor;\n \n // If still too large after base scaling, scale down further\n if (scaledWidth > targetMaxWidth || scaledHeight > targetMaxHeight) {\n const scaleX = targetMaxWidth / scaledWidth;\n const scaleY = targetMaxHeight / scaledHeight;\n scaleFactor *= Math.min(scaleX, scaleY);\n }\n \n width *= scaleFactor;\n height *= scaleFactor;\n \n // Use higher DPR for crisp rendering at smaller sizes\n const dpr = 2; // Fixed 2x for consistent quality\n canvas.width = width * dpr;\n canvas.height = height * dpr;\n canvas.style.width = width + 'px';\n canvas.style.height = height + 'px';\n \n const ctx = canvas.getContext('2d');\n ctx.scale(dpr, dpr);\n \n // White background\n ctx.fillStyle = \"#FFFFFF\";\n ctx.fillRect(0, 0, width, height);\n \n // Draw the SVG image at logical size\n ctx.drawImage(img, 0, 0, width, height);\n \n // Clean up URL\n URL.revokeObjectURL(url);\n \n // Return data URL\n resolve(canvas.toDataURL('image/png'));\n };\n \n img.onerror = () => {\n URL.revokeObjectURL(url);\n reject(new Error('Failed to load SVG image'));\n };\n \n img.src = url;\n });\n \n // Replace math element with img tag containing the PNG data URL\n const imgElement = document.createElement('img');\n imgElement.src = dataUrl;\n \n // Extract dimensions from the data URL canvas\n const img2 = new Image();\n img2.src = dataUrl;\n await new Promise((resolve) => {\n img2.onload = resolve;\n img2.onerror = resolve;\n setTimeout(resolve, 100); // Timeout fallback\n });\n \n // Set explicit dimensions (accounting for DPR)\n const displayWidth = img2.naturalWidth / 2; // Divide by DPR\n const displayHeight = img2.naturalHeight / 2;\n \n imgElement.width = displayWidth;\n imgElement.height = displayHeight;\n imgElement.style.cssText = `display:inline-block;margin:0.5em 0;width:${displayWidth}px;height:${displayHeight}px;vertical-align:middle;`;\n imgElement.alt = 'Math equation';\n \n mathEl.parentNode.replaceChild(imgElement, mathEl);\n } catch (error) {\n console.error('Failed to convert math element to image:', error);\n // Keep the original element if conversion fails\n }\n }\n }\n \n // 2. Process GeoJSON maps - convert to static images (following Gem's guide)\n const geojsonContainers = clone.querySelectorAll('.geojson-container');\n if (geojsonContainers.length > 0) {\n \n for (const clonedContainer of geojsonContainers) {\n try {\n // Find the corresponding live container by matching data-original-source\n const originalSource = clonedContainer.getAttribute('data-original-source');\n if (!originalSource) {\n console.warn('No original source found for GeoJSON container');\n continue;\n }\n \n // Find live container with same source\n let liveContainer = null;\n const allLiveContainers = previewPanel.querySelectorAll('.geojson-container');\n for (const candidate of allLiveContainers) {\n if (candidate.getAttribute('data-original-source') === originalSource) {\n liveContainer = candidate;\n break;\n }\n }\n \n if (!liveContainer) {\n console.warn('Could not find live GeoJSON container');\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Interactive content not available in copy]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n continue;\n }\n \n // Check if map is ready\n const map = liveContainer._map;\n if (!map) {\n console.warn('Map not initialized yet');\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Still loading]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n continue;\n }\n \n // Rasterize the map to PNG\n const dataUrl = await rasterizeGeoJSONMap(liveContainer);\n \n if (dataUrl) {\n // Replace with image\n const img = document.createElement('img');\n img.src = dataUrl;\n img.style.cssText = 'width: 100%; height: 300px; border: 1px solid #ddd; border-radius: 4px; margin: 0.5em 0;';\n img.alt = 'GeoJSON Map';\n clonedContainer.parentNode.replaceChild(img, clonedContainer);\n } else {\n // Fallback placeholder\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Interactive content not available in copy]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n }\n \n } catch (error) {\n console.error('Failed to process GeoJSON container:', error);\n // Replace with placeholder\n const placeholder = document.createElement('div');\n placeholder.style.cssText = 'padding: 12px; background-color: #f0f0f0; border: 1px solid #ccc; text-align: center; margin: 0.5em 0; border-radius: 4px;';\n placeholder.textContent = '[GeoJSON Map - Interactive content not available in copy]';\n clonedContainer.parentNode.replaceChild(placeholder, clonedContainer);\n }\n }\n }\n \n \n \n // 6. Process GeoJSON/Leaflet maps - capture as single image (compose tiles + overlays)\n const mapContainers = clone.querySelectorAll('[data-qd-lang=\"geojson\"]');\n for (const container of mapContainers) {\n try {\n const containerId = container.id;\n const originalContainer = containerId ? previewPanel.querySelector(`#${containerId}`) : null;\n if (!originalContainer) continue;\n const leafletContainer = originalContainer.querySelector('.leaflet-container');\n if (!leafletContainer) continue;\n\n const dpr = Math.max(1, window.devicePixelRatio || 1);\n const width = leafletContainer.clientWidth || 600;\n const height = leafletContainer.clientHeight || 400;\n const canvas = document.createElement('canvas');\n canvas.width = Math.round(width * dpr);\n canvas.height = Math.round(height * dpr);\n const ctx = canvas.getContext('2d');\n ctx.scale(dpr, dpr);\n ctx.fillStyle = '#FFFFFF';\n ctx.fillRect(0, 0, width, height);\n\n const leafRect = leafletContainer.getBoundingClientRect();\n\n // Draw tiles (snap to integer pixels to avoid seams)\n const tiles = Array.from(leafletContainer.querySelectorAll('img.leaflet-tile'));\n for (const tile of tiles) {\n try {\n const r = tile.getBoundingClientRect();\n const x = Math.round(r.left - leafRect.left);\n const y = Math.round(r.top - leafRect.top);\n const w = Math.round(r.width);\n const h = Math.round(r.height);\n const overlaps = !(r.right <= leafRect.left || r.left >= leafRect.right || r.bottom <= leafRect.top || r.top >= leafRect.bottom);\n const style = window.getComputedStyle(tile);\n if (w > 0 && h > 0 && overlaps && style.display !== 'none' && style.visibility !== 'hidden') {\n ctx.drawImage(tile, x, y, w + 1, h + 1);\n }\n } catch (e) {\n console.warn('Failed to draw tile:', e);\n }\n }\n\n // Draw SVG overlays (paths, markers)\n const overlaySvgs = originalContainer.querySelectorAll('.leaflet-overlay-pane svg');\n for (const svg of overlaySvgs) {\n try {\n const svgStr = new XMLSerializer().serializeToString(svg);\n const dataUrl = 'data:image/svg+xml;charset=utf-8,' + encodeURIComponent(svgStr);\n const img = new Image();\n await new Promise((resolve) => { img.onload = resolve; img.onerror = resolve; img.src = dataUrl; });\n const r = svg.getBoundingClientRect();\n const x = Math.round(r.left - leafRect.left);\n const y = Math.round(r.top - leafRect.top);\n const w = Math.round(r.width);\n const h = Math.round(r.height);\n const overlaps = !(r.right <= leafRect.left || r.left >= leafRect.right || r.bottom <= leafRect.top || r.top >= leafRect.bottom);\n if (w > 0 && h > 0 && overlaps) ctx.drawImage(img, x, y, w, h);\n } catch (e) {\n console.warn('Failed to draw overlay SVG:', e);\n }\n }\n\n // Draw marker icons (PNG/SVG img elements)\n const markerIcons = originalContainer.querySelectorAll('.leaflet-marker-pane img.leaflet-marker-icon');\n for (const icon of markerIcons) {\n try {\n const r = icon.getBoundingClientRect();\n const x = Math.round(r.left - leafRect.left);\n const y = Math.round(r.top - leafRect.top);\n const w = Math.round(r.width);\n const h = Math.round(r.height);\n const overlaps = !(r.right <= leafRect.left || r.left >= leafRect.right || r.bottom <= leafRect.top || r.top >= leafRect.bottom);\n const style = window.getComputedStyle(icon);\n if (w > 0 && h > 0 && overlaps && style.display !== 'none' && style.visibility !== 'hidden') {\n ctx.drawImage(icon, x, y, w, h);\n }\n } catch (e) {\n console.warn('Failed to draw marker icon:', e);\n }\n }\n\n // Try to produce a data URL (may fail if canvas tainted by CORS tiles)\n let mapDataUrl = '';\n try {\n mapDataUrl = canvas.toDataURL('image/png', 1.0);\n } catch (_e) {\n console.warn('Map canvas tainted; falling back to placeholder');\n }\n\n const img = document.createElement('img');\n if (mapDataUrl) {\n img.src = mapDataUrl;\n img.width = width;\n img.height = height;\n img.setAttribute('width', String(width));\n img.setAttribute('height', String(height));\n img.style.width = width + 'px';\n img.style.height = height + 'px';\n img.style.display = 'block';\n img.style.border = '1px solid #ddd';\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n img.alt = 'Map';\n } else {\n img.alt = 'Map';\n img.style.width = width + 'px';\n img.style.height = height + 'px';\n img.style.border = '1px solid #ddd';\n img.style.backgroundColor = '#f0f0f0';\n }\n\n container.parentNode.replaceChild(img, container);\n } catch (err) {\n console.warn('Failed to process map container:', err);\n }\n }\n \n // 7. Process HTML fence blocks - render the HTML content and process images\n const htmlContainers = clone.querySelectorAll('.qde-html-container');\n for (const container of htmlContainers) {\n try {\n // Get the original source HTML\n const source = container.getAttribute('data-qd-source');\n \n // Check if there's a pre element (fallback display) or actual HTML content\n const pre = container.querySelector('pre');\n \n if (source) {\n // Parse the source HTML\n const tempDiv = document.createElement('div');\n tempDiv.innerHTML = source;\n \n // Process all images in the HTML block\n const htmlImages = tempDiv.querySelectorAll('img');\n for (const img of htmlImages) {\n // Preserve original dimensions from HTML attributes\n const widthAttr = img.getAttribute('width');\n const heightAttr = img.getAttribute('height');\n \n if (widthAttr) {\n img.width = parseInt(widthAttr);\n img.style.width = widthAttr.includes('%') ? widthAttr : `${img.width}px`;\n }\n if (heightAttr) {\n img.height = parseInt(heightAttr);\n img.style.height = heightAttr.includes('%') ? heightAttr : `${img.height}px`;\n }\n \n // Convert to data URL using canvas (like squibview)\n if (img.src && !img.src.startsWith('data:')) {\n try {\n // Use canvas to convert image to data URL (avoids CORS issues)\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n \n // Create new image and wait for it to load\n const tempImg = new Image();\n tempImg.crossOrigin = 'anonymous';\n \n await new Promise((resolve, reject) => {\n tempImg.onload = function() {\n \n // Calculate dimensions preserving aspect ratio\n let displayWidth = 0;\n let displayHeight = 0;\n \n // Use the width specified in HTML (e.g. width=\"250\")\n if (widthAttr && !widthAttr.includes('%')) {\n displayWidth = parseInt(widthAttr);\n }\n \n // Use the height if specified\n if (heightAttr && !heightAttr.includes('%')) {\n displayHeight = parseInt(heightAttr);\n }\n \n \n // If only width is specified, calculate height based on aspect ratio\n if (displayWidth > 0 && displayHeight === 0) {\n if (tempImg.naturalWidth > 0) {\n const aspectRatio = tempImg.naturalHeight / tempImg.naturalWidth;\n displayHeight = Math.round(displayWidth * aspectRatio);\n }\n }\n // If only height is specified, calculate width based on aspect ratio\n else if (displayHeight > 0 && displayWidth === 0) {\n if (tempImg.naturalHeight > 0) {\n const aspectRatio = tempImg.naturalWidth / tempImg.naturalHeight;\n displayWidth = Math.round(displayHeight * aspectRatio);\n }\n }\n // If neither specified, use natural dimensions\n else if (displayWidth === 0 && displayHeight === 0) {\n displayWidth = tempImg.naturalWidth || 250;\n displayHeight = tempImg.naturalHeight || 200;\n }\n \n \n canvas.width = displayWidth;\n canvas.height = displayHeight;\n \n // Draw image to canvas\n ctx.drawImage(tempImg, 0, 0, displayWidth, displayHeight);\n \n // Convert to data URL\n const dataUrl = canvas.toDataURL('image/png', 1.0);\n \n // Update original image\n img.src = dataUrl;\n img.width = displayWidth;\n img.height = displayHeight;\n img.setAttribute('width', displayWidth.toString());\n img.setAttribute('height', displayHeight.toString());\n img.style.width = displayWidth + 'px';\n img.style.height = displayHeight + 'px';\n \n resolve();\n };\n \n tempImg.onerror = function() {\n console.warn('Failed to load HTML fence image:', img.src);\n reject(new Error('Image load failed'));\n };\n \n // Set source - resolve relative paths\n if (img.src.startsWith('http') || img.src.startsWith('//')) {\n tempImg.src = img.src;\n } else {\n // Relative path - let browser resolve it\n const absoluteImg = new Image();\n absoluteImg.src = img.src;\n tempImg.src = absoluteImg.src;\n }\n });\n } catch (err) {\n console.warn('Failed to convert HTML fence image:', img.src, err);\n }\n }\n \n // Add v:shapes for Word compatibility\n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n }\n \n // Replace container content with processed HTML (whether it had pre or not)\n container.innerHTML = tempDiv.innerHTML;\n } else if (!pre) {\n // Container has rendered HTML already, process its images directly\n const htmlImages = container.querySelectorAll('img');\n for (const img of htmlImages) {\n // Same image processing as above\n const widthAttr = img.getAttribute('width');\n const heightAttr = img.getAttribute('height');\n \n if (widthAttr) {\n img.width = parseInt(widthAttr);\n img.style.width = widthAttr.includes('%') ? widthAttr : `${img.width}px`;\n }\n if (heightAttr) {\n img.height = parseInt(heightAttr);\n img.style.height = heightAttr.includes('%') ? heightAttr : `${img.height}px`;\n }\n \n if (img.src && !img.src.startsWith('data:')) {\n try {\n // Use same canvas approach as above\n const canvas = document.createElement('canvas');\n const ctx = canvas.getContext('2d');\n const tempImg = new Image();\n tempImg.crossOrigin = 'anonymous';\n \n await new Promise((resolve, reject) => {\n tempImg.onload = function() {\n // Calculate dimensions preserving aspect ratio\n let displayWidth = img.width || 0;\n let displayHeight = img.height || 0;\n \n // If only width is specified, calculate height based on aspect ratio\n if (displayWidth && !displayHeight) {\n const aspectRatio = tempImg.naturalHeight / tempImg.naturalWidth;\n displayHeight = Math.round(displayWidth * aspectRatio);\n }\n // If only height is specified, calculate width based on aspect ratio\n else if (displayHeight && !displayWidth) {\n const aspectRatio = tempImg.naturalWidth / tempImg.naturalHeight;\n displayWidth = Math.round(displayHeight * aspectRatio);\n }\n // If neither specified, use natural dimensions\n else if (!displayWidth && !displayHeight) {\n displayWidth = tempImg.naturalWidth || 250;\n displayHeight = tempImg.naturalHeight || Math.round(250 * (tempImg.naturalHeight / tempImg.naturalWidth));\n }\n \n canvas.width = displayWidth;\n canvas.height = displayHeight;\n ctx.drawImage(tempImg, 0, 0, displayWidth, displayHeight);\n \n const dataUrl = canvas.toDataURL('image/png', 1.0);\n img.src = dataUrl;\n img.width = displayWidth;\n img.height = displayHeight;\n img.setAttribute('width', displayWidth.toString());\n img.setAttribute('height', displayHeight.toString());\n img.style.width = displayWidth + 'px';\n img.style.height = displayHeight + 'px';\n \n resolve();\n };\n \n tempImg.onerror = function() {\n console.warn('Failed to load HTML fence image:', img.src);\n reject(new Error('Image load failed'));\n };\n \n if (img.src.startsWith('http') || img.src.startsWith('//')) {\n tempImg.src = img.src;\n } else {\n const absoluteImg = new Image();\n absoluteImg.src = img.src;\n tempImg.src = absoluteImg.src;\n }\n });\n } catch (err) {\n console.warn('Failed to convert HTML fence image:', img.src, err);\n }\n }\n \n img.setAttribute('v:shapes', 'image' + Math.random().toString(36).substr(2, 9));\n }\n }\n } catch (err) {\n console.warn('Failed to process HTML container:', err);\n }\n }\n \n // 8. Tables are already HTML tables from the built-in renderer\n // No processing needed\n \n // Wrap in proper HTML structure for rich text editors\n const fragment = clone.innerHTML;\n const htmlContent = `\n <!DOCTYPE html>\n <html xmlns:v=\"urn:schemas-microsoft-com:vml\"\n xmlns:o=\"urn:schemas-microsoft-com:office:office\"\n xmlns:w=\"urn:schemas-microsoft-com:office:word\">\n <head>\n <meta charset=\"utf-8\">\n <style>\n /* Table styling */\n table { border-collapse: collapse; width: 100%; margin-bottom: 1em; }\n th, td { border: 1px solid #ccc; padding: 8px; text-align: left; }\n th { background-color: #f0f0f0; font-weight: bold; }\n \n /* Code block styling */\n pre { background-color: #f4f4f4; padding: 1em; border-radius: 4px; overflow-x: auto; }\n code { font-family: monospace; background-color: #f4f4f4; padding: 0.2em 0.4em; border-radius: 3px; }\n \n /* Image handling */\n img { display: block; max-width: 100%; height: auto; margin: 0.5em 0; }\n \n /* Blockquote */\n blockquote { border-left: 4px solid #ddd; margin-left: 0; padding-left: 1em; color: #666; }\n \n /* Math equations centered like squibview */\n .math-display { text-align: center; margin: 1em 0; }\n .math-display img { display: inline-block; margin: 0 auto; }\n </style>\n </head>\n <body><!--StartFragment-->${fragment}<!--EndFragment--></body>\n </html>`;\n \n // Get plain text version\n const text = clone.textContent || clone.innerText || '';\n \n // Get platform for clipboard strategy (like squibview)\n const platform = getPlatform();\n \n if (platform === 'macos') {\n // macOS approach (like squibview)\n try {\n await navigator.clipboard.write([\n new ClipboardItem({\n 'text/html': new Blob([htmlContent], { type: 'text/html' }),\n 'text/plain': new Blob([text], { type: 'text/plain' })\n })\n ]);\n return { success: true, html: htmlContent, text };\n } catch (modernErr) {\n console.warn('Modern clipboard API failed, trying Safari fallback:', modernErr);\n // Safari fallback (selection-based HTML of fragment)\n if (copyToClipboard(fragment)) {\n return { success: true, html: htmlContent, text };\n }\n throw new Error('Fallback copy failed');\n }\n } else {\n // Windows/Linux approach (like squibview)\n const tempDiv = document.createElement('div');\n tempDiv.style.position = 'fixed';\n tempDiv.style.left = '-9999px';\n tempDiv.style.top = '0';\n // Use fragment for selection-based fallback copy\n tempDiv.innerHTML = fragment;\n document.body.appendChild(tempDiv);\n \n try {\n await navigator.clipboard.write([\n new ClipboardItem({\n 'text/html': new Blob([htmlContent], { type: 'text/html' }),\n 'text/plain': new Blob([text], { type: 'text/plain' })\n })\n ]);\n return { success: true, html: htmlContent, text };\n } catch (modernErr) {\n console.warn('Modern clipboard API failed, trying execCommand fallback:', modernErr);\n const selection = window.getSelection();\n const range = document.createRange();\n range.selectNodeContents(tempDiv);\n selection.removeAllRanges();\n selection.addRange(range);\n \n const successful = document.execCommand('copy');\n if (!successful) {\n throw new Error('Fallback copy failed');\n }\n return { success: true, html: htmlContent, text };\n } finally {\n if (tempDiv && tempDiv.parentNode) {\n document.body.removeChild(tempDiv);\n }\n }\n }\n \n } catch (err) {\n console.error('Failed to copy rendered content:', err);\n throw err;\n }\n}\n\n","/**\n * Quikdown Editor - A drop-in markdown editor control\n * @version 1.0.5\n * @license BSD-2-Clause\n */\n\nimport quikdown_bd from './quikdown_bd.js';\nimport { getRenderedContent } from './quikdown_edit_copy.js';\n\n// Default options\nconst DEFAULT_OPTIONS = {\n mode: 'split', // 'source' | 'preview' | 'split'\n showToolbar: true,\n showRemoveHR: false, // Show button to remove horizontal rules (---)\n showLazyLinefeeds: false, // Show button to convert lazy linefeeds\n theme: 'auto', // 'light' | 'dark' | 'auto'\n lazy_linefeeds: false,\n inline_styles: false, // Use CSS classes (false) or inline styles (true)\n debounceDelay: 20, // Reduced from 100ms for better responsiveness\n placeholder: 'Start typing markdown...',\n plugins: {\n highlightjs: false,\n mermaid: false\n },\n /**\n * Preload fence-rendering libraries at construction time so the FIRST\n * encounter with a fence type renders instantly (no lazy load delay).\n *\n * Accepts:\n * - 'all' — preload every known library\n * - ['highlightjs','mermaid','math',\n * 'geojson','stl'] — preload specific libraries\n * - [{ name: 'mylib', script: 'https://...', css: '...' }]\n * — preload an arbitrary library\n *\n * Without this, fence libraries are loaded on demand the first time their\n * fence type is encountered. That keeps the editor lightweight, but the\n * first SVG/Mermaid/Math/GeoJSON/STL fence will show \"loading...\" for a\n * moment. Set `preloadFences` if you want zero-delay rendering — at the\n * cost of a few hundred KB of upfront network.\n *\n * Developer's choice. The editor itself is still ~70 KB minified;\n * `preloadFences` only affects the OPTIONAL fence renderers.\n */\n preloadFences: null,\n customFences: {}, // { 'language': (code, lang) => html }\n enableComplexFences: true, // Enable CSV tables, math rendering, SVG, etc.\n showUndoRedo: false, // Show undo/redo toolbar buttons\n undoStackSize: 100 // Maximum number of undo states to keep\n};\n\n// Library catalog used by preloadFences. Each entry knows how to:\n// - check if the library is already on the page (so we don't double-load)\n// - load it via script (and optional CSS)\nconst FENCE_LIBRARIES = {\n highlightjs: {\n check: () => typeof window.hljs !== 'undefined',\n script: 'https://unpkg.com/@highlightjs/cdn-assets/highlight.min.js',\n css: 'https://unpkg.com/@highlightjs/cdn-assets/styles/github.min.css',\n cssDark: 'https://unpkg.com/@highlightjs/cdn-assets/styles/github-dark.min.css'\n },\n mermaid: {\n check: () => typeof window.mermaid !== 'undefined',\n script: 'https://unpkg.com/mermaid/dist/mermaid.min.js',\n afterLoad: () => {\n if (window.mermaid) window.mermaid.initialize({ startOnLoad: false });\n }\n },\n math: {\n check: () => typeof window.MathJax !== 'undefined',\n script: 'https://cdn.jsdelivr.net/npm/mathjax@3/es5/tex-svg.js',\n beforeLoad: () => {\n // Configure MathJax before loading (must be set on window before script runs)\n if (!window.MathJax) {\n window.MathJax = {\n tex: { inlineMath: [['$', '$'], ['\\\\(', '\\\\)']], displayMath: [['$$', '$$'], ['\\\\[', '\\\\]']] },\n svg: { fontCache: 'global' },\n startup: { typeset: false }\n };\n }\n }\n },\n geojson: {\n check: () => typeof window.L !== 'undefined',\n script: 'https://unpkg.com/leaflet@1.9.4/dist/leaflet.js',\n css: 'https://unpkg.com/leaflet@1.9.4/dist/leaflet.css'\n },\n stl: {\n check: () => typeof window.THREE !== 'undefined',\n script: 'https://unpkg.com/three@0.147.0/build/three.min.js'\n }\n};\n\n/**\n * Quikdown Editor - A complete markdown editing solution\n */\nclass QuikdownEditor {\n constructor(container, options = {}) {\n // Resolve container\n this.container = typeof container === 'string' \n ? document.querySelector(container) \n : container;\n \n if (!this.container) {\n throw new Error('QuikdownEditor: Invalid container');\n }\n \n // Merge options\n this.options = { ...DEFAULT_OPTIONS, ...options };\n \n // State\n this._markdown = '';\n this._html = '';\n this.currentMode = this.options.mode;\n this.updateTimer = null;\n\n // Undo/redo state\n this._undoStack = [];\n this._redoStack = [];\n this._isUndoRedo = false;\n \n // Initialize\n this.initPromise = this.init();\n }\n \n /**\n * Initialize the editor\n */\n async init() {\n // Load plugins if requested\n await this.loadPlugins();\n \n // Build UI\n this.buildUI();\n \n // Attach event listeners\n this.attachEvents();\n \n // Apply initial theme\n this.applyTheme();\n \n // Set initial mode\n this.setMode(this.currentMode);\n \n // Set initial content if provided\n if (this.options.initialContent) {\n this.setMarkdown(this.options.initialContent);\n }\n }\n \n /**\n * Build the editor UI\n */\n buildUI() {\n // Clear container\n this.container.innerHTML = '';\n \n // Add editor class\n this.container.classList.add('qde-container');\n \n // Create toolbar if enabled\n if (this.options.showToolbar) {\n this.toolbar = this.createToolbar();\n this.container.appendChild(this.toolbar);\n }\n \n // Create editor area\n this.editorArea = document.createElement('div');\n this.editorArea.className = 'qde-editor';\n \n // Create source panel\n this.sourcePanel = document.createElement('div');\n this.sourcePanel.className = 'qde-source';\n \n this.sourceTextarea = document.createElement('textarea');\n this.sourceTextarea.className = 'qde-textarea';\n this.sourceTextarea.spellcheck = false;\n this.sourceTextarea.placeholder = this.options.placeholder;\n this.sourcePanel.appendChild(this.sourceTextarea);\n \n // Create preview panel\n this.previewPanel = document.createElement('div');\n this.previewPanel.className = 'qde-preview';\n this.previewPanel.contentEditable = true;\n this.previewPanel.spellcheck = false;\n \n // Add panels to editor\n this.editorArea.appendChild(this.sourcePanel);\n this.editorArea.appendChild(this.previewPanel);\n this.container.appendChild(this.editorArea);\n \n // Add built-in styles if not already present\n this.injectStyles();\n }\n \n /**\n * Create toolbar\n */\n createToolbar() {\n const toolbar = document.createElement('div');\n toolbar.className = 'qde-toolbar';\n \n // Mode buttons\n const modes = ['source', 'split', 'preview'];\n const modeLabels = { source: 'Source', split: 'Split', preview: 'Rendered' };\n modes.forEach(mode => {\n const btn = document.createElement('button');\n btn.className = 'qde-btn';\n btn.dataset.mode = mode;\n btn.textContent = modeLabels[mode];\n btn.title = `Switch to ${modeLabels[mode]} view`;\n toolbar.appendChild(btn);\n });\n \n // Undo/Redo buttons (if enabled)\n if (this.options.showUndoRedo) {\n const undoBtn = document.createElement('button');\n undoBtn.className = 'qde-btn disabled';\n undoBtn.dataset.action = 'undo';\n undoBtn.textContent = 'Undo';\n undoBtn.title = 'Undo (Ctrl+Z)';\n toolbar.appendChild(undoBtn);\n\n const redoBtn = document.createElement('button');\n redoBtn.className = 'qde-btn disabled';\n redoBtn.dataset.action = 'redo';\n redoBtn.textContent = 'Redo';\n redoBtn.title = 'Redo (Ctrl+Shift+Z / Ctrl+Y)';\n toolbar.appendChild(redoBtn);\n }\n\n // Spacer\n const spacer = document.createElement('span');\n spacer.className = 'qde-spacer';\n toolbar.appendChild(spacer);\n \n // Copy buttons\n const copyButtons = [\n { action: 'copy-markdown', text: 'Copy MD', title: 'Copy markdown to clipboard' },\n { action: 'copy-html', text: 'Copy HTML', title: 'Copy HTML to clipboard' },\n { action: 'copy-rendered', text: 'Copy Rendered', title: 'Copy rich text to clipboard' }\n ];\n \n copyButtons.forEach(({ action, text, title }) => {\n const btn = document.createElement('button');\n btn.className = 'qde-btn';\n btn.dataset.action = action;\n btn.textContent = text;\n btn.title = title;\n toolbar.appendChild(btn);\n });\n \n // Remove HR button (if enabled)\n if (this.options.showRemoveHR) {\n const removeHRBtn = document.createElement('button');\n removeHRBtn.className = 'qde-btn';\n removeHRBtn.dataset.action = 'remove-hr';\n removeHRBtn.textContent = 'Remove HR';\n removeHRBtn.title = 'Remove all horizontal rules (---) from markdown';\n toolbar.appendChild(removeHRBtn);\n }\n\n // Lazy linefeeds button (if enabled)\n if (this.options.showLazyLinefeeds) {\n const lazyLFBtn = document.createElement('button');\n lazyLFBtn.className = 'qde-btn';\n lazyLFBtn.dataset.action = 'lazy-linefeeds';\n lazyLFBtn.textContent = 'Fix Linefeeds';\n lazyLFBtn.title = 'Convert single newlines to paragraph breaks (one-time transform)';\n toolbar.appendChild(lazyLFBtn);\n }\n \n return toolbar;\n }\n \n /**\n * Inject built-in styles\n */\n injectStyles() {\n if (document.getElementById('qde-styles')) return;\n \n const style = document.createElement('style');\n style.id = 'qde-styles';\n style.textContent = `\n .qde-container {\n display: flex;\n flex-direction: column;\n height: 100%;\n border: 1px solid #ddd;\n border-radius: 4px;\n overflow: hidden;\n background: white;\n }\n \n .qde-toolbar {\n display: flex;\n align-items: center;\n padding: 8px;\n background: #f5f5f5;\n border-bottom: 1px solid #ddd;\n gap: 4px;\n }\n \n .qde-btn {\n padding: 6px 12px;\n border: 1px solid #ccc;\n background: white;\n border-radius: 3px;\n cursor: pointer;\n font-size: 14px;\n transition: all 0.2s;\n }\n \n .qde-btn:hover {\n background: #e9e9e9;\n border-color: #999;\n }\n \n .qde-btn.active {\n background: #007bff;\n color: white;\n border-color: #0056b3;\n }\n\n .qde-btn.disabled {\n opacity: 0.4;\n pointer-events: none;\n }\n \n .qde-spacer {\n flex: 1;\n }\n \n .qde-editor {\n display: flex;\n flex: 1;\n overflow: hidden;\n }\n \n .qde-source, .qde-preview {\n flex: 1 1 0;\n min-width: 0; /* allow flex shrinking below content size */\n min-height: 0;\n overflow: auto;\n padding: 16px;\n box-sizing: border-box;\n }\n\n .qde-source {\n border-right: 1px solid #ddd;\n /* Source pane is just a container for the textarea — make it\n a positioning context so the textarea can fill it absolutely */\n position: relative;\n padding: 0; /* textarea brings its own padding */\n }\n\n .qde-textarea {\n display: block;\n position: absolute;\n inset: 0;\n width: 100%;\n height: 100%;\n border: none;\n outline: none;\n resize: none;\n padding: 16px;\n box-sizing: border-box;\n font-family: 'Monaco', 'Courier New', monospace;\n font-size: 14px;\n line-height: 1.5;\n background: transparent;\n color: inherit;\n /* Wrap long lines so the textarea only scrolls VERTICALLY.\n pre-wrap preserves intentional line breaks/whitespace\n while soft-wrapping at the right edge. */\n white-space: pre-wrap;\n word-wrap: break-word;\n overflow-x: hidden;\n overflow-y: auto;\n }\n \n .qde-preview {\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;\n font-size: 16px;\n line-height: 1.6;\n outline: none;\n cursor: text; /* Standard text cursor */\n overflow-x: hidden; /* never scroll horizontally; clip wide content */\n }\n\n /* Code blocks and inline code — self-contained so the editor\n does not depend on any external stylesheet for these. */\n .qde-preview pre {\n background: #f4f4f4;\n color: #1f2937;\n padding: 10px;\n border-radius: 4px;\n overflow-x: auto;\n margin: 0.6em 0;\n font-size: 0.9em;\n line-height: 1.5;\n font-family: ui-monospace, \"SF Mono\", Monaco, \"Cascadia Code\",\n \"Roboto Mono\", Consolas, \"Courier New\", monospace;\n }\n .qde-preview code {\n padding: 2px 4px;\n font-size: 0.9em;\n border-radius: 3px;\n background: #f0f0f0;\n color: #1f2937;\n font-family: ui-monospace, \"SF Mono\", Monaco, \"Cascadia Code\",\n \"Roboto Mono\", Consolas, \"Courier New\", monospace;\n }\n .qde-preview pre code {\n padding: 0;\n font-size: inherit;\n border-radius: 0;\n background: transparent;\n color: inherit;\n }\n\n /* Wide fence content (Leaflet maps, large SVGs, STL canvases,\n iframes, raw <img>) must never overflow the preview pane */\n .qde-preview .geojson-container,\n .qde-preview .qde-stl-container,\n .qde-preview .qde-svg-container,\n .qde-preview .leaflet-container,\n .qde-preview iframe,\n .qde-preview img,\n .qde-preview > svg {\n max-width: 100%;\n }\n .qde-preview .leaflet-container { box-sizing: border-box; }\n\n /* Standard markdown tables (the .quikdown-table class) need to\n scroll horizontally inside their own wrapper rather than\n making the whole preview pane scroll */\n .qde-preview table.quikdown-table,\n .qde-preview table.qde-csv-table {\n display: block;\n max-width: 100%;\n overflow-x: auto;\n }\n\n /* Fence-specific styles */\n .qde-svg-container {\n max-width: 100%;\n overflow: auto;\n }\n\n .qde-svg-container svg {\n max-width: 100%;\n height: auto;\n }\n \n .qde-html-container {\n /* HTML containers inherit background */\n margin: 12px 0;\n }\n \n .qde-math-container {\n text-align: center;\n margin: 16px 0;\n overflow-x: auto;\n }\n \n /* All tables in preview (both regular markdown and CSV) */\n .qde-preview table {\n width: 100%;\n border-collapse: collapse;\n margin: 12px 0;\n font-size: 14px;\n }\n \n .qde-preview table th,\n .qde-preview table td {\n border: 1px solid #ddd;\n padding: 8px;\n }\n \n /* Support for alignment classes from quikdown */\n .qde-preview .quikdown-left { text-align: left; }\n .qde-preview .quikdown-center { text-align: center; }\n .qde-preview .quikdown-right { text-align: right; }\n \n .qde-preview table th {\n background: #f5f5f5;\n font-weight: bold;\n }\n \n .qde-preview table tr:nth-child(even) {\n background: #f9f9f9;\n }\n \n /* Specific to CSV-generated tables */\n .qde-data-table {\n /* Can add specific CSV table styles here if needed */\n }\n \n .qde-json {\n /* Let highlight.js handle styling */\n overflow-x: auto;\n }\n \n .qde-error {\n background: #fee;\n border: 1px solid #fcc;\n color: #c00;\n padding: 8px;\n border-radius: 4px;\n font-family: monospace;\n font-size: 12px;\n }\n \n /* Read-only complex fence blocks in preview */\n .qde-preview [contenteditable=\"false\"] {\n cursor: auto; /* Use automatic cursor (arrow for non-text) */\n user-select: text;\n position: relative;\n }\n \n /* Reset headings inside the preview to plain browser defaults so\n parent-page styles (site navs, marketing pages, design systems)\n cannot bleed in. Business-casual: black text, decreasing sizes,\n no decorative borders. See docs/quikdown-editor.md for how\n embedders can override these with their own stylesheet. */\n .qde-preview h1 { font-size: 2em; }\n .qde-preview h2 { font-size: 1.5em; }\n .qde-preview h3 { font-size: 1.25em; }\n .qde-preview h4 { font-size: 1em; }\n .qde-preview h5 { font-size: 0.875em; }\n .qde-preview h6 { font-size: 0.85em; }\n .qde-preview h1,\n .qde-preview h2,\n .qde-preview h3,\n .qde-preview h4,\n .qde-preview h5,\n .qde-preview h6 {\n font-weight: bold;\n color: inherit;\n border: none;\n margin: 0.6em 0 0.3em 0;\n line-height: 1.25;\n }\n .qde-preview p {\n margin: 0.35em 0;\n }\n .qde-preview ul,\n .qde-preview ol {\n padding-left: 1.8em;\n margin: 0.4em 0;\n }\n .qde-preview li {\n margin: 0.15em 0;\n }\n .qde-preview blockquote {\n margin: 0.5em 0;\n padding-left: 1em;\n }\n\n /* Ensure proper cursor for editable text elements */\n .qde-preview p,\n .qde-preview h1,\n .qde-preview h2,\n .qde-preview h3,\n .qde-preview h4,\n .qde-preview h5,\n .qde-preview h6,\n .qde-preview li,\n .qde-preview td,\n .qde-preview th,\n .qde-preview blockquote,\n .qde-preview pre[contenteditable=\"true\"],\n .qde-preview code[contenteditable=\"true\"] {\n cursor: text;\n }\n \n \n /* Non-editable complex renderers */\n .qde-preview .qde-svg-container[contenteditable=\"false\"],\n .qde-preview .qde-html-container[contenteditable=\"false\"],\n .qde-preview .qde-math-container[contenteditable=\"false\"],\n .qde-preview .mermaid[contenteditable=\"false\"] {\n opacity: 0.98;\n }\n \n /* Subtle hover effect for read-only blocks */\n .qde-preview [contenteditable=\"false\"]:hover::after {\n content: \"Read-only\";\n position: absolute;\n top: 2px;\n right: 2px;\n font-size: 10px;\n color: #999;\n background: rgba(255, 255, 255, 0.9);\n padding: 2px 4px;\n border-radius: 2px;\n pointer-events: none;\n }\n \n /* Fix list padding in preview */\n .qde-preview ul,\n .qde-preview ol {\n padding-left: 2em;\n margin: 0.5em 0;\n }\n \n .qde-preview li {\n margin: 0.25em 0;\n }\n \n /* Mode-specific visibility */\n .qde-mode-source .qde-preview { display: none; }\n .qde-mode-source .qde-source { border-right: none; }\n .qde-mode-preview .qde-source { display: none; }\n .qde-mode-split .qde-source,\n .qde-mode-split .qde-preview { display: block; }\n \n /* Dark theme */\n .qde-dark {\n background: #1e1e1e;\n color: #e0e0e0;\n border-color: #444;\n }\n \n .qde-dark .qde-toolbar {\n background: #2d2d2d;\n border-color: #444;\n }\n \n .qde-dark .qde-btn {\n background: #3a3a3a;\n color: #e0e0e0;\n border-color: #555;\n }\n \n .qde-dark .qde-btn:hover {\n background: #4a4a4a;\n }\n \n .qde-dark .qde-source {\n border-color: #444;\n }\n \n .qde-dark .qde-textarea {\n background: #1e1e1e;\n color: #e0e0e0;\n }\n \n .qde-dark .qde-preview {\n background: #1e1e1e;\n color: #e0e0e0;\n }\n \n /* Dark mode code blocks */\n .qde-dark .qde-preview pre {\n background: #2d2d3a;\n color: #e6e6f0;\n }\n .qde-dark .qde-preview code {\n background: #2a2a3a;\n color: #e6e6f0;\n }\n .qde-dark .qde-preview pre code {\n background: transparent;\n color: inherit;\n }\n\n /* Dark mode table styles */\n .qde-dark .qde-preview table th,\n .qde-dark .qde-preview table td {\n border-color: #3a3a3a;\n }\n \n .qde-dark .qde-preview table th {\n background: #2d2d2d;\n }\n \n .qde-dark .qde-preview table tr:nth-child(even) {\n background: #252525;\n }\n \n /* Mobile responsive */\n @media (max-width: 768px) {\n .qde-mode-split .qde-editor {\n flex-direction: column;\n }\n\n .qde-mode-split .qde-source {\n border-right: none;\n border-bottom: 1px solid #ddd;\n }\n .qde-dark.qde-mode-split .qde-source {\n border-bottom-color: #444;\n }\n }\n `;\n \n document.head.appendChild(style);\n }\n \n /**\n * Attach event listeners\n */\n attachEvents() {\n // Source textarea input\n this.sourceTextarea.addEventListener('input', () => {\n this.handleSourceInput();\n });\n \n // Preview contenteditable input\n this.previewPanel.addEventListener('input', () => {\n this.handlePreviewInput();\n });\n \n // Toolbar buttons\n if (this.toolbar) {\n this.toolbar.addEventListener('click', (e) => {\n const btn = e.target.closest('.qde-btn');\n if (!btn) return;\n \n if (btn.dataset.mode) {\n this.setMode(btn.dataset.mode);\n } else if (btn.dataset.action) {\n this.handleAction(btn.dataset.action);\n }\n });\n }\n \n // Keyboard shortcuts\n document.addEventListener('keydown', (e) => {\n if (e.ctrlKey || e.metaKey) {\n switch(e.key) {\n case '1':\n e.preventDefault();\n this.setMode('source');\n break;\n case '2':\n e.preventDefault();\n this.setMode('split');\n break;\n case '3':\n e.preventDefault();\n this.setMode('preview');\n break;\n case 'z':\n case 'Z':\n if (e.shiftKey) {\n e.preventDefault();\n this.redo();\n } else {\n e.preventDefault();\n this.undo();\n }\n break;\n case 'y':\n case 'Y':\n e.preventDefault();\n this.redo();\n break;\n }\n }\n });\n }\n \n /**\n * Handle source textarea input\n */\n handleSourceInput() {\n clearTimeout(this.updateTimer);\n this.updateTimer = setTimeout(() => {\n this.updateFromMarkdown(this.sourceTextarea.value);\n }, this.options.debounceDelay);\n }\n \n /**\n * Handle preview panel input\n */\n handlePreviewInput() {\n clearTimeout(this.updateTimer);\n this.updateTimer = setTimeout(() => {\n this.updateFromHTML();\n }, this.options.debounceDelay);\n }\n \n /**\n * Update from markdown source\n */\n updateFromMarkdown(markdown) {\n // Push current state to undo stack before changing (unless this is an undo/redo operation)\n if (!this._isUndoRedo) {\n this._pushUndoState(markdown || '');\n }\n this._isUndoRedo = false;\n\n this._markdown = markdown || '';\n \n // Show placeholder if empty\n if (!this._markdown.trim()) {\n this._html = '';\n if (this.currentMode !== 'source') {\n this.previewPanel.innerHTML = '<div style=\"color: #999; font-style: italic; padding: 16px;\">Start typing markdown in the source panel...</div>';\n }\n } else {\n this._html = quikdown_bd(markdown, {\n fence_plugin: this.createFencePlugin(),\n lazy_linefeeds: this.options.lazy_linefeeds,\n inline_styles: this.options.inline_styles\n });\n \n // Update preview if visible\n if (this.currentMode !== 'source') {\n this.previewPanel.innerHTML = this._html;\n // Make all fence blocks non-editable\n this.makeFencesNonEditable();\n \n // Process all math elements with MathJax if loaded (like squibview)\n if (window.MathJax && window.MathJax.typesetPromise) {\n const mathElements = this.previewPanel.querySelectorAll('.math-display');\n if (mathElements.length > 0) {\n window.MathJax.typesetPromise(Array.from(mathElements))\n .catch(_err => {\n console.warn('MathJax batch processing failed:', _err);\n });\n }\n }\n }\n }\n \n // Trigger change event\n if (this.options.onChange) {\n this.options.onChange(this._markdown, this._html);\n }\n }\n \n /**\n * Update from HTML preview\n */\n updateFromHTML() {\n // Clone the preview panel to avoid modifying the actual DOM\n const clonedPanel = this.previewPanel.cloneNode(true);\n\n // Pre-process special elements on the clone\n this.preprocessSpecialElements(clonedPanel);\n\n this._html = this.previewPanel.innerHTML;\n const newMarkdown = quikdown_bd.toMarkdown(clonedPanel, {\n fence_plugin: this.createFencePlugin()\n });\n\n // Push previous state to undo stack (now that we know the new markdown)\n if (!this._isUndoRedo) {\n this._pushUndoState(newMarkdown);\n }\n this._isUndoRedo = false;\n\n this._markdown = newMarkdown;\n\n // Update source if visible\n if (this.currentMode !== 'preview') {\n this.sourceTextarea.value = this._markdown;\n }\n\n // Trigger change event\n if (this.options.onChange) {\n this.options.onChange(this._markdown, this._html);\n }\n\n this._updateUndoButtons();\n }\n \n /**\n * Pre-process special elements before markdown conversion\n */\n preprocessSpecialElements(panel) {\n if (!panel) return;\n \n // Restore non-editable complex fences from their data attributes\n const complexFences = panel.querySelectorAll('[contenteditable=\"false\"][data-qd-source]');\n complexFences.forEach(element => {\n const source = element.getAttribute('data-qd-source');\n const fence = element.getAttribute('data-qd-fence') || '```';\n const lang = element.getAttribute('data-qd-lang') || '';\n \n // Create a pre element with the original source\n const pre = document.createElement('pre');\n pre.setAttribute('data-qd-fence', fence);\n if (lang) pre.setAttribute('data-qd-lang', lang);\n const code = document.createElement('code');\n // The source is already the original unescaped content when using setAttribute\n // No need to unescape since browser handles it automatically\n code.textContent = source;\n pre.appendChild(code);\n \n // Replace the complex element with pre\n element.parentNode.replaceChild(pre, element);\n });\n \n // Convert CSV tables back to CSV fence blocks (these ARE editable)\n const csvTables = panel.querySelectorAll('table.qde-csv-table[data-qd-lang]');\n csvTables.forEach(table => {\n const lang = table.getAttribute('data-qd-lang');\n if (!lang || !['csv', 'psv', 'tsv'].includes(lang)) return;\n \n const delimiter = lang === 'csv' ? ',' : lang === 'psv' ? '|' : '\\t';\n \n // Extract data from table\n let csv = '';\n \n // Get headers\n const headers = [];\n const headerCells = table.querySelectorAll('thead th');\n headerCells.forEach(th => {\n const text = th.textContent.trim();\n // Quote if contains delimiter or quotes\n const needsQuoting = text.includes(delimiter) || text.includes('\"') || text.includes('\\n');\n headers.push(needsQuoting ? `\"${text.replace(/\"/g, '\"\"')}\"` : text);\n });\n csv += headers.join(delimiter) + '\\n';\n \n // Get rows\n const rows = table.querySelectorAll('tbody tr');\n rows.forEach(tr => {\n const cells = [];\n tr.querySelectorAll('td').forEach(td => {\n const text = td.textContent.trim();\n const needsQuoting = text.includes(delimiter) || text.includes('\"') || text.includes('\\n');\n cells.push(needsQuoting ? `\"${text.replace(/\"/g, '\"\"')}\"` : text);\n });\n csv += cells.join(delimiter) + '\\n';\n });\n \n // Create a pre element with the CSV data\n const pre = document.createElement('pre');\n pre.setAttribute('data-qd-fence', '```');\n pre.setAttribute('data-qd-lang', lang);\n const code = document.createElement('code');\n code.textContent = csv.trim();\n pre.appendChild(code);\n \n // Replace table with pre\n table.parentNode.replaceChild(pre, table);\n });\n }\n \n /**\n * Create fence plugin for syntax highlighting\n */\n createFencePlugin() {\n const render = (code, lang) => {\n // Check custom fences first (they take precedence)\n if (this.options.customFences && this.options.customFences[lang]) {\n try {\n return this.options.customFences[lang](code, lang);\n } catch (err) {\n console.error(`Custom fence plugin error for ${lang}:`, err);\n return `<pre><code class=\"language-${lang}\">${this.escapeHtml(code)}</code></pre>`;\n }\n }\n \n // For bidirectional editing, only apply syntax highlighting\n // Skip complex transformations that break round-trip conversion\n const skipComplexRendering = !this.options.enableComplexFences;\n \n if (!skipComplexRendering) {\n // Built-in lazy loading fence handlers (disabled for now)\n switch(lang) {\n case 'svg':\n return this.renderSVG(code);\n \n case 'html':\n return this.renderHTML(code);\n \n case 'math':\n case 'tex':\n case 'latex':\n return this.renderMath(code, lang);\n \n case 'csv':\n case 'psv':\n case 'tsv':\n return this.renderTable(code, lang);\n \n case 'json':\n case 'json5':\n return this.renderJSON(code, lang);\n \n case 'katex': // Use MathJax for katex fence blocks (backward compatibility)\n return this.renderMath(code, 'katex');\n \n case 'mermaid':\n if (window.mermaid) {\n return this.renderMermaid(code);\n }\n break;\n \n case 'geojson':\n return this.renderGeoJSON(code);\n \n case 'stl':\n return this.renderSTL(code);\n }\n }\n \n // Syntax highlighting support - keep editable for bidirectional\n if (window.hljs && lang && hljs.getLanguage(lang)) {\n const highlighted = hljs.highlight(code, { language: lang }).value;\n // Don't add contenteditable=\"false\" - the bidirectional system can extract text from the highlighted code\n return `<pre data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\"><code class=\"hljs language-${lang}\">${highlighted}</code></pre>`;\n }\n \n // Default: let quikdown handle it\n return undefined;\n };\n \n // Reverse function to extract raw source from rendered HTML\n const reverse = (element) => {\n // Get the language from data attribute\n const lang = element.getAttribute('data-qd-lang') || '';\n let content = '';\n \n // For syntax-highlighted code, extract the raw text\n if (element.querySelector('code.hljs')) {\n const code = element.querySelector('code.hljs');\n content = code.textContent || code.innerText || '';\n }\n // For other code blocks, just get the text content\n else if (element.querySelector('code')) {\n const codeEl = element.querySelector('code');\n content = codeEl.textContent || codeEl.innerText || '';\n }\n // Fallback to element text\n else {\n content = element.textContent || element.innerText || '';\n }\n \n // Return in the format quikdown_bd expects\n return {\n content: content,\n lang: lang,\n fence: '```'\n };\n };\n \n // Return object format for v1.1.0 API with both render and reverse\n return { render, reverse };\n }\n \n /**\n * Render SVG content\n */\n renderSVG(code) {\n try {\n // Basic SVG validation\n const parser = new DOMParser();\n const doc = parser.parseFromString(code, 'image/svg+xml');\n const parseError = doc.querySelector('parsererror');\n \n if (parseError) {\n throw new Error('Invalid SVG');\n }\n \n // Sanitize SVG by removing script tags and event handlers\n const svg = doc.documentElement;\n svg.querySelectorAll('script').forEach(el => el.remove());\n \n // Remove event handlers\n const walker = document.createTreeWalker(svg, NodeFilter.SHOW_ELEMENT);\n let node;\n while ((node = walker.nextNode())) {\n for (let i = node.attributes.length - 1; i >= 0; i--) {\n const attr = node.attributes[i];\n if (attr.name.startsWith('on') || attr.value.includes('javascript:')) {\n node.removeAttribute(attr.name);\n }\n }\n }\n \n // Create container element programmatically to avoid attribute escaping issues\n const container = document.createElement('div');\n container.className = 'qde-svg-container';\n container.contentEditable = 'false';\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'svg');\n container.setAttribute('data-qd-source', code); // No escaping needed when using setAttribute!\n container.innerHTML = new XMLSerializer().serializeToString(svg);\n \n // Return the HTML string\n return container.outerHTML;\n } catch (err) {\n const errorContainer = document.createElement('pre');\n errorContainer.className = 'qde-error';\n errorContainer.contentEditable = 'false';\n errorContainer.setAttribute('data-qd-fence', '```');\n errorContainer.setAttribute('data-qd-lang', 'svg');\n errorContainer.textContent = `Invalid SVG: ${err.message}`;\n return errorContainer.outerHTML;\n }\n }\n \n /**\n * Render HTML content with DOMPurify if available\n */\n renderHTML(code) {\n const id = `html-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n \n // If DOMPurify is loaded, use it\n if (window.DOMPurify) {\n const clean = DOMPurify.sanitize(code);\n \n // Create container programmatically\n const container = document.createElement('div');\n container.className = 'qde-html-container';\n container.contentEditable = 'false';\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'html');\n container.setAttribute('data-qd-source', code);\n container.innerHTML = clean;\n \n return container.outerHTML;\n }\n \n // Try to lazy load DOMPurify\n this.lazyLoadLibrary(\n 'DOMPurify',\n () => window.DOMPurify,\n 'https://unpkg.com/dompurify/dist/purify.min.js'\n ).then(loaded => {\n if (loaded) {\n const element = document.getElementById(id);\n if (element) {\n const clean = DOMPurify.sanitize(code);\n element.innerHTML = clean;\n // Update attributes after loading\n element.setAttribute('data-qd-source', code);\n element.setAttribute('data-qd-fence', '```');\n element.setAttribute('data-qd-lang', 'html');\n }\n }\n });\n \n // Return placeholder with bidirectional attributes - non-editable\n const placeholder = document.createElement('div');\n placeholder.id = id;\n placeholder.className = 'qde-html-container';\n placeholder.contentEditable = 'false';\n placeholder.setAttribute('data-qd-fence', '```');\n placeholder.setAttribute('data-qd-lang', 'html');\n placeholder.setAttribute('data-qd-source', code);\n const pre = document.createElement('pre');\n pre.textContent = code;\n placeholder.appendChild(pre);\n \n return placeholder.outerHTML;\n }\n \n /**\n * Render math with MathJax (SVG output for better copy support)\n */\n renderMath(code, _lang) {\n const id = `math-${Math.random().toString(36).substring(2, 15)}`;\n \n // Create container exactly like squibview\n const container = document.createElement('div');\n container.id = id;\n container.className = 'math-display';\n container.contentEditable = 'false';\n container.setAttribute('data-source-type', 'math');\n \n // Format content for MathJax (display mode with $$) - exactly like squibview\n const singleLineContent = code.replace(/\\r?\\n/g, ' ').replace(/\\s+/g, ' ').trim();\n container.textContent = `$$${singleLineContent}$$`;\n \n // Add centering style\n container.style.textAlign = 'center';\n container.style.margin = '1em 0';\n \n \n // Ensure MathJax will be loaded (if not already)\n if (!window.MathJax || !window.MathJax.typesetPromise) {\n this.ensureMathJaxLoaded();\n }\n \n // MathJax will be processed in batch after preview update\n return container.outerHTML;\n }\n \n /**\n * Ensures MathJax is loaded (but doesn't process elements)\n */\n ensureMathJaxLoaded() {\n if (typeof window.MathJax === 'undefined' && !window.mathJaxLoading) {\n window.mathJaxLoading = true;\n \n // Configure MathJax before loading\n if (!window.MathJax) {\n window.MathJax = {\n loader: { load: ['input/tex', 'output/svg'] },\n tex: { \n packages: { '[+]': ['ams'] },\n inlineMath: [['$', '$'], ['\\\\(', '\\\\)']],\n displayMath: [['$$', '$$'], ['\\\\[', '\\\\]']],\n processEscapes: true,\n processEnvironments: true\n },\n options: {\n renderActions: { addMenu: [] },\n ignoreHtmlClass: 'tex2jax_ignore',\n processHtmlClass: 'tex2jax_process'\n },\n svg: {\n fontCache: 'none' // Important: self-contained SVGs for copy\n },\n startup: { typeset: false }\n };\n }\n \n const script = document.createElement('script');\n script.src = 'https://cdn.jsdelivr.net/npm/mathjax@3.2.2/es5/tex-svg.js';\n script.async = true;\n script.onload = () => {\n window.mathJaxLoading = false;\n \n // Process any existing math elements (like squibview)\n if (window.MathJax && window.MathJax.typesetPromise) {\n const mathElements = document.querySelectorAll('.math-display');\n if (mathElements.length > 0) {\n window.MathJax.typesetPromise(Array.from(mathElements)).catch(err => {\n console.warn('Initial MathJax processing failed:', err);\n });\n }\n }\n };\n script.onerror = () => {\n window.mathJaxLoading = false;\n console.error('Failed to load MathJax');\n };\n document.head.appendChild(script);\n }\n }\n \n /**\n * Render CSV/PSV/TSV as HTML table\n */\n renderTable(code, lang) {\n const escapedCode = this.escapeHtml(code);\n try {\n const delimiter = lang === 'csv' ? ',' : lang === 'psv' ? '|' : '\\t';\n const lines = code.trim().split('\\n');\n \n if (lines.length === 0) {\n return `<pre data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\" data-qd-source=\"${escapedCode}\">${escapedCode}</pre>`;\n }\n \n // CSV tables CAN be editable - we'll convert HTML table back to CSV\n // Don't need data-qd-source since we convert the table structure back to CSV\n let html = `<table class=\"qde-data-table qde-csv-table\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\">`;\n \n // Parse header\n const header = this.parseCSVLine(lines[0], delimiter);\n html += '<thead><tr>';\n header.forEach(cell => {\n html += `<th>${this.escapeHtml(cell.trim())}</th>`;\n });\n html += '</tr></thead>';\n \n // Parse body\n if (lines.length > 1) {\n html += '<tbody>';\n for (let i = 1; i < lines.length; i++) {\n const row = this.parseCSVLine(lines[i], delimiter);\n html += '<tr>';\n row.forEach(cell => {\n html += `<td>${this.escapeHtml(cell.trim())}</td>`;\n });\n html += '</tr>';\n }\n html += '</tbody>';\n }\n \n html += '</table>';\n return html;\n } catch (_err) {\n return `<pre data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\" data-qd-source=\"${escapedCode}\">${escapedCode}</pre>`;\n }\n }\n\n /**\n * Parse CSV line handling quoted values\n */\n parseCSVLine(line, delimiter) {\n const result = [];\n let current = '';\n let inQuotes = false;\n \n for (let i = 0; i < line.length; i++) {\n const char = line[i];\n const nextChar = line[i + 1];\n \n if (char === '\"') {\n if (inQuotes && nextChar === '\"') {\n current += '\"';\n i++; // Skip next quote\n } else {\n inQuotes = !inQuotes;\n }\n } else if (char === delimiter && !inQuotes) {\n result.push(current);\n current = '';\n } else {\n current += char;\n }\n }\n \n result.push(current);\n return result;\n }\n \n /**\n * Render JSON with syntax highlighting\n */\n renderJSON(code, lang) {\n // If highlight.js is available, use it for all JSON\n if (window.hljs && hljs.getLanguage('json')) {\n try {\n // Try to format if valid JSON\n let toHighlight = code;\n try {\n const data = JSON.parse(code);\n toHighlight = JSON.stringify(data, null, 2);\n } catch (_e) {\n // Use original if not valid JSON\n }\n \n const highlighted = hljs.highlight(toHighlight, { language: 'json' }).value;\n return `<pre class=\"qde-json\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\"><code class=\"hljs language-json\">${highlighted}</code></pre>`;\n } catch (_e) {\n // Fall through if highlighting fails\n }\n }\n \n // No highlighting available - return plain\n return `<pre class=\"qde-json\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"${lang}\">${this.escapeHtml(code)}</pre>`;\n }\n \n /**\n * Render GeoJSON map\n */\n renderGeoJSON(code) {\n // Generate unique map ID (following SquibView pattern)\n const mapId = `map-${Math.random().toString(36).substr(2, 15)}`;\n \n // Function to render the map\n const renderMap = () => {\n const container = document.getElementById(mapId + '-container');\n if (!container || !window.L) return;\n \n try {\n const data = JSON.parse(code);\n \n // Clear container and set deterministic size for rasterization\n const mapDiv = document.createElement('div');\n mapDiv.id = mapId;\n mapDiv.style.cssText = 'width: 100%; height: 300px;';\n container.innerHTML = '';\n container.appendChild(mapDiv);\n \n // Create the map\n const map = L.map(mapId);\n \n // Store back-reference for capture (per Gem's guide)\n container._map = map; // Avoid window pollution\n \n // Add tile layer with CORS support\n const tileLayer = L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {\n attribution: '',\n crossOrigin: 'anonymous' // Important for canvas capture\n });\n tileLayer.addTo(map);\n \n // Add GeoJSON layer\n const geoJsonLayer = L.geoJSON(data);\n geoJsonLayer.addTo(map);\n \n // Fit bounds if valid\n if (geoJsonLayer.getBounds().isValid()) {\n map.fitBounds(geoJsonLayer.getBounds());\n } else {\n map.setView([0, 0], 2);\n }\n \n // Store references for copy-time capture\n container._tileLayer = tileLayer;\n container._geoJsonLayer = geoJsonLayer;\n \n // Optional: Wait for tiles to load for better capture\n tileLayer.on('load', () => {\n container.setAttribute('data-tiles-loaded', 'true');\n });\n \n } catch (err) {\n container.innerHTML = `<pre class=\"qde-error\">GeoJSON error: ${this.escapeHtml(err.message)}</pre>`;\n }\n };\n \n // Check if Leaflet is already loaded\n if (window.L) {\n // Render after DOM update\n setTimeout(renderMap, 0);\n } else {\n // Lazy load Leaflet only if not already loading\n if (!window._qde_leaflet_loading) {\n window._qde_leaflet_loading = this.lazyLoadLibrary(\n 'Leaflet',\n () => window.L,\n 'https://unpkg.com/leaflet@1.9.4/dist/leaflet.js',\n 'https://unpkg.com/leaflet@1.9.4/dist/leaflet.css'\n ).catch(err => {\n console.warn('Failed to load Leaflet:', err);\n // Clear the loading promise so it can be retried\n window._qde_leaflet_loading = null;\n return false;\n });\n }\n \n window._qde_leaflet_loading.then(loaded => {\n if (loaded) {\n renderMap();\n } else {\n const element = document.getElementById(mapId + '-container');\n if (element) {\n element.innerHTML = '<div style=\"padding: 20px; text-align: center; color: #666;\">Failed to load map library</div>';\n }\n }\n }).catch(() => {\n // Error already handled above\n });\n }\n \n // Return container following SquibView pattern\n const container = document.createElement('div');\n container.className = 'geojson-container';\n container.id = mapId + '-container';\n container.style.cssText = 'width: 100%; height: 300px; border: 1px solid #ddd; border-radius: 4px; margin: 0.5em 0; background: #f0f0f0;';\n container.contentEditable = 'false';\n \n // Preserve source for copy-time identification (per Gem's guide)\n container.setAttribute('data-source-type', 'geojson');\n container.setAttribute('data-original-source', this.escapeHtml(code));\n \n // For bidirectional editing\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'geojson');\n container.setAttribute('data-qd-source', code);\n \n container.textContent = 'Loading map...';\n \n return container.outerHTML;\n }\n \n /**\n * Render STL 3D model\n */\n renderSTL(code) {\n const id = `qde-stl-viewer-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n\n // Function to render the 3D model (assumes window.THREE is loaded)\n const render3D = () => {\n const element = document.getElementById(id);\n if (!element) return;\n\n try {\n const THREE = window.THREE;\n \n // Create scene\n const scene = new THREE.Scene();\n scene.background = new THREE.Color(0xf0f0f0);\n \n // Create camera\n const camera = new THREE.PerspectiveCamera(75, element.clientWidth / 400, 0.1, 1000);\n \n // Create renderer\n const renderer = new THREE.WebGLRenderer({ antialias: true });\n renderer.setSize(element.clientWidth, 400);\n element.innerHTML = '';\n element.appendChild(renderer.domElement);\n \n // Store Three.js references for copy functionality (like squibview)\n element._threeScene = scene;\n element._threeCamera = camera;\n element._threeRenderer = renderer;\n \n // Parse STL data (ASCII format)\n const geometry = this.parseSTL(code);\n const material = new THREE.MeshLambertMaterial({ color: 0x0066ff });\n const mesh = new THREE.Mesh(geometry, material);\n scene.add(mesh);\n \n // Add lighting\n const ambientLight = new THREE.AmbientLight(0x404040, 0.6);\n scene.add(ambientLight);\n \n const directionalLight = new THREE.DirectionalLight(0xffffff, 0.8);\n directionalLight.position.set(1, 1, 1).normalize();\n scene.add(directionalLight);\n \n // Position camera based on object bounds\n const box = new THREE.Box3().setFromObject(mesh);\n const center = box.getCenter(new THREE.Vector3());\n const size = box.getSize(new THREE.Vector3());\n const maxDim = Math.max(size.x, size.y, size.z);\n \n camera.position.set(center.x + maxDim, center.y + maxDim, center.z + maxDim);\n camera.lookAt(center);\n \n // Animate\n const animate = () => {\n requestAnimationFrame(animate);\n mesh.rotation.y += 0.01;\n renderer.render(scene, camera);\n };\n animate();\n } catch (err) {\n console.error('STL rendering error:', err);\n element.innerHTML = `<pre class=\"qde-error\">STL error: ${this.escapeHtml(err.message)}</pre>`;\n }\n };\n \n // If Three.js is already loaded, render immediately. Otherwise lazy-load\n // it from a CDN (matches the GeoJSON/Leaflet pattern).\n if (window.THREE) {\n setTimeout(render3D, 0);\n } else {\n if (!window._qde_three_loading) {\n window._qde_three_loading = this.lazyLoadLibrary(\n 'Three.js',\n () => window.THREE,\n 'https://unpkg.com/three@0.147.0/build/three.min.js'\n ).catch(_err => {\n console.warn('Failed to load Three.js for STL rendering');\n window._qde_three_loading = null;\n return false;\n });\n }\n window._qde_three_loading.then(loaded => {\n if (loaded) {\n render3D();\n } else {\n const element = document.getElementById(id);\n if (element) {\n element.innerHTML = '<div style=\"padding: 20px; text-align: center; color: #666;\">Failed to load Three.js for STL rendering</div>';\n }\n }\n });\n }\n\n // Return placeholder with data-stl-id for copy functionality\n return `<div id=\"${id}\" class=\"qde-stl-container\" data-stl-id=\"${id}\" data-qd-fence=\"\\`\\`\\`\" data-qd-lang=\"stl\" data-qd-source=\"${this.escapeHtml(code)}\" contenteditable=\"false\" style=\"height: 400px; background: #f0f0f0; display: flex; align-items: center; justify-content: center;\">Loading 3D model...</div>`;\n }\n \n /**\n * Parse ASCII STL format\n * @param {string} stlData - The STL file content\n * @returns {THREE.BufferGeometry} - The parsed geometry\n */\n parseSTL(stlData) {\n const THREE = window.THREE;\n const geometry = new THREE.BufferGeometry();\n const vertices = [];\n const normals = [];\n \n const lines = stlData.split('\\n');\n let currentNormal = null;\n \n for (let line of lines) {\n line = line.trim();\n \n if (line.startsWith('facet normal')) {\n const parts = line.split(/\\s+/);\n currentNormal = [parseFloat(parts[2]), parseFloat(parts[3]), parseFloat(parts[4])];\n } else if (line.startsWith('vertex')) {\n const parts = line.split(/\\s+/);\n vertices.push(parseFloat(parts[1]), parseFloat(parts[2]), parseFloat(parts[3]));\n if (currentNormal) {\n normals.push(currentNormal[0], currentNormal[1], currentNormal[2]);\n }\n }\n }\n \n geometry.setAttribute('position', new THREE.Float32BufferAttribute(vertices, 3));\n geometry.setAttribute('normal', new THREE.Float32BufferAttribute(normals, 3));\n \n return geometry;\n }\n \n /**\n * Render Mermaid diagram\n */\n renderMermaid(code) {\n const id = `mermaid-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n setTimeout(() => {\n const element = document.getElementById(id);\n if (element && window.mermaid) {\n mermaid.render(id + '-svg', code).then(result => {\n element.innerHTML = result.svg;\n }).catch(err => {\n element.innerHTML = `<pre>Error rendering diagram: ${err.message}</pre>`;\n });\n }\n }, 0);\n \n // Create container programmatically\n const container = document.createElement('div');\n container.id = id;\n container.className = 'mermaid';\n container.contentEditable = 'false';\n container.setAttribute('data-qd-source', code);\n container.setAttribute('data-qd-fence', '```');\n container.setAttribute('data-qd-lang', 'mermaid');\n container.textContent = 'Loading diagram...';\n \n return container.outerHTML;\n }\n \n /**\n * Escape HTML for attributes\n */\n escapeHtml(text) {\n return (text ?? \"\").replace(/[&\"'<>]/g, m => \n ({'&':'&','\"':'"',\"'\":''','<':'<','>':'>'}[m]));\n }\n \n /**\n * Make complex fence blocks non-editable\n */\n makeFencesNonEditable() {\n if (!this.previewPanel) return;\n \n // Only make specific complex fence types non-editable\n // SVG, HTML, Math, Mermaid already have contenteditable=\"false\" set\n // Syntax-highlighted code also has it set\n \n // Don't make regular code blocks or tables non-editable\n // They can be edited and properly round-trip\n }\n \n /**\n * Load plugins dynamically — honors both `plugins: { highlightjs, mermaid }`\n * (legacy) and the newer `preloadFences` option which can preload any\n * combination of fence libraries (or 'all') at construction time.\n */\n async loadPlugins() {\n const namesToLoad = new Set();\n\n // Legacy plugins option\n if (this.options.plugins) {\n if (this.options.plugins.highlightjs) namesToLoad.add('highlightjs');\n if (this.options.plugins.mermaid) namesToLoad.add('mermaid');\n }\n\n // New preloadFences option\n const pf = this.options.preloadFences;\n if (pf === 'all') {\n Object.keys(FENCE_LIBRARIES).forEach(n => namesToLoad.add(n));\n } else if (Array.isArray(pf)) {\n for (const entry of pf) {\n if (typeof entry === 'string') {\n if (FENCE_LIBRARIES[entry]) namesToLoad.add(entry);\n else console.warn(`QuikdownEditor: unknown preloadFences entry \"${entry}\"`);\n } else if (entry && typeof entry === 'object' && entry.script) {\n // Custom library: { name, script, css? }\n namesToLoad.add('__custom__:' + (entry.name || entry.script));\n FENCE_LIBRARIES['__custom__:' + (entry.name || entry.script)] = {\n check: () => false,\n script: entry.script,\n css: entry.css\n };\n }\n }\n } else if (pf) {\n console.warn('QuikdownEditor: preloadFences should be \"all\", an array, or null');\n }\n\n // Load each in parallel; respect already-loaded state\n const promises = [];\n for (const name of namesToLoad) {\n const lib = FENCE_LIBRARIES[name];\n if (!lib || lib.check()) continue;\n if (lib.beforeLoad) lib.beforeLoad();\n const p = (async () => {\n try {\n const tasks = [];\n if (lib.script) tasks.push(this.loadScript(lib.script));\n if (lib.css) tasks.push(this.loadCSS(lib.css, 'qde-hljs-light'));\n if (lib.cssDark) tasks.push(this.loadCSS(lib.cssDark, 'qde-hljs-dark'));\n await Promise.all(tasks);\n if (lib.css && lib.cssDark) this._syncHljsTheme();\n if (lib.afterLoad) lib.afterLoad();\n } catch (err) {\n console.warn(`QuikdownEditor: failed to preload ${name}:`, err);\n }\n })();\n promises.push(p);\n }\n\n await Promise.all(promises);\n }\n \n /**\n * Lazy load library if not already loaded\n */\n async lazyLoadLibrary(name, check, scriptUrl, cssUrl = null) {\n // Check if library is already loaded\n if (check()) {\n return true;\n }\n \n try {\n const promises = [];\n \n // Load script\n if (scriptUrl) {\n promises.push(this.loadScript(scriptUrl));\n }\n \n // Load CSS if provided\n if (cssUrl) {\n promises.push(this.loadCSS(cssUrl));\n }\n \n await Promise.all(promises);\n \n // Verify library loaded\n return check();\n } catch (err) {\n console.error(`Failed to load ${name}:`, err);\n return false;\n }\n }\n \n /**\n * Load external script\n */\n loadScript(src) {\n return new Promise((resolve, reject) => {\n const script = document.createElement('script');\n script.src = src;\n script.onload = resolve;\n script.onerror = reject;\n document.head.appendChild(script);\n });\n }\n \n /**\n * Load external CSS\n */\n loadCSS(href, id) {\n return new Promise((resolve) => {\n const link = document.createElement('link');\n link.rel = 'stylesheet';\n link.href = href;\n if (id) link.id = id;\n link.onload = resolve;\n document.head.appendChild(link);\n // Resolve anyway after timeout (CSS doesn't always fire onload)\n setTimeout(resolve, 1000);\n });\n }\n\n /**\n * Enable the hljs stylesheet matching the current theme and disable\n * the other one. Called from applyTheme and after hljs CSS loads.\n */\n _syncHljsTheme() {\n const isDark = this.container.classList.contains('qde-dark');\n const light = document.getElementById('qde-hljs-light');\n const dark = document.getElementById('qde-hljs-dark');\n if (light) light.disabled = isDark;\n if (dark) dark.disabled = !isDark;\n }\n\n /**\n * Apply the current theme (based on this.options.theme)\n */\n applyTheme() {\n const theme = this.options.theme;\n\n // Tear down any previous auto-mode listener so we don't stack them\n if (this._autoThemeListener) {\n window.matchMedia('(prefers-color-scheme: dark)').removeEventListener('change', this._autoThemeListener);\n this._autoThemeListener = null;\n }\n\n if (theme === 'auto') {\n const mq = window.matchMedia('(prefers-color-scheme: dark)');\n this.container.classList.toggle('qde-dark', mq.matches);\n this._autoThemeListener = (e) => {\n this.container.classList.toggle('qde-dark', e.matches);\n this._syncHljsTheme();\n };\n mq.addEventListener('change', this._autoThemeListener);\n } else {\n this.container.classList.toggle('qde-dark', theme === 'dark');\n }\n this._syncHljsTheme();\n }\n\n /**\n * Set theme at runtime. Accepts 'light', 'dark', or 'auto'.\n * @param {'light'|'dark'|'auto'} theme\n */\n setTheme(theme) {\n if (!['light', 'dark', 'auto'].includes(theme)) return;\n this.options.theme = theme;\n this.applyTheme();\n }\n\n /**\n * Get the current theme option (as configured, not resolved).\n * @returns {'light'|'dark'|'auto'}\n */\n getTheme() {\n return this.options.theme;\n }\n \n /**\n * Set lazy linefeeds option\n * @param {boolean} enabled - Whether to enable lazy linefeeds\n */\n setLazyLinefeeds(enabled) {\n this.options.lazy_linefeeds = enabled;\n // Re-render if we have content\n if (this._markdown) {\n this.updateFromMarkdown(this._markdown);\n }\n }\n \n /**\n * Get lazy linefeeds option\n * @returns {boolean}\n */\n getLazyLinefeeds() {\n return this.options.lazy_linefeeds;\n }\n \n /**\n * Set debounce delay for input updates\n * @param {number} delay - Delay in milliseconds (0 for instant)\n */\n setDebounceDelay(delay) {\n this.options.debounceDelay = Math.max(0, delay);\n }\n \n /**\n * Get current debounce delay\n * @returns {number} Delay in milliseconds\n */\n getDebounceDelay() {\n return this.options.debounceDelay;\n }\n \n /**\n * Set editor mode\n */\n setMode(mode) {\n if (!['source', 'preview', 'split'].includes(mode)) return;\n\n // Preserve theme class across mode swap (the assignment to className\n // below would otherwise wipe it out — this used to be a no-op bug\n // where dark mode was lost on every setMode call).\n const wasDark = this.container.classList.contains('qde-dark');\n\n this.currentMode = mode;\n this.container.className = `qde-container qde-mode-${mode}`;\n if (wasDark) {\n this.container.classList.add('qde-dark');\n }\n\n // Update toolbar buttons\n if (this.toolbar) {\n this.toolbar.querySelectorAll('.qde-btn[data-mode]').forEach(btn => {\n btn.classList.toggle('active', btn.dataset.mode === mode);\n });\n }\n \n // Make fence blocks non-editable when showing preview\n if (mode !== 'source') {\n setTimeout(() => this.makeFencesNonEditable(), 0);\n }\n \n // Trigger mode change event\n if (this.options.onModeChange) {\n this.options.onModeChange(mode);\n }\n }\n \n // --- Undo / Redo ---\n\n /**\n * Push current markdown state onto the undo stack (called before a change).\n * Only pushes if the new state differs from the current state.\n * @param {string} newMarkdown - the incoming markdown (used to detect no-op)\n * @private\n */\n _pushUndoState(newMarkdown) {\n // Don't push if the content hasn't actually changed\n if (newMarkdown === this._markdown) return;\n\n this._undoStack.push(this._markdown);\n\n // Enforce max stack size\n const max = this.options.undoStackSize || 100;\n if (this._undoStack.length > max) {\n this._undoStack.splice(0, this._undoStack.length - max);\n }\n\n // Any new edit clears the redo stack\n this._redoStack = [];\n this._updateUndoButtons();\n }\n\n /**\n * Undo the last change. Restores the previous markdown state.\n */\n undo() {\n if (!this.canUndo()) return;\n // Save current state to redo stack\n this._redoStack.push(this._markdown);\n const previous = this._undoStack.pop();\n this._isUndoRedo = true;\n // Update state directly (setMarkdown is async; keep it synchronous here)\n this._markdown = previous;\n if (this.sourceTextarea) {\n this.sourceTextarea.value = previous;\n }\n this.updateFromMarkdown(previous);\n this._updateUndoButtons();\n }\n\n /**\n * Redo the last undone change.\n */\n redo() {\n if (!this.canRedo()) return;\n // Save current state to undo stack\n this._undoStack.push(this._markdown);\n const next = this._redoStack.pop();\n this._isUndoRedo = true;\n this._markdown = next;\n if (this.sourceTextarea) {\n this.sourceTextarea.value = next;\n }\n this.updateFromMarkdown(next);\n this._updateUndoButtons();\n }\n\n /**\n * @returns {boolean} true if undo is possible\n */\n canUndo() {\n return this._undoStack.length > 0;\n }\n\n /**\n * @returns {boolean} true if redo is possible\n */\n canRedo() {\n return this._redoStack.length > 0;\n }\n\n /**\n * Clear the undo and redo history.\n */\n clearHistory() {\n this._undoStack = [];\n this._redoStack = [];\n this._updateUndoButtons();\n }\n\n /**\n * Update the disabled state of the undo/redo toolbar buttons.\n * @private\n */\n _updateUndoButtons() {\n if (!this.toolbar) return;\n const undoBtn = this.toolbar.querySelector('[data-action=\"undo\"]');\n const redoBtn = this.toolbar.querySelector('[data-action=\"redo\"]');\n if (undoBtn) {\n undoBtn.classList.toggle('disabled', !this.canUndo());\n }\n if (redoBtn) {\n redoBtn.classList.toggle('disabled', !this.canRedo());\n }\n }\n\n /**\n * Handle toolbar actions\n */\n handleAction(action) {\n switch(action) {\n case 'copy-markdown':\n this.copy('markdown');\n break;\n case 'copy-html':\n this.copy('html');\n break;\n case 'copy-rendered':\n this.copyRendered();\n break;\n case 'remove-hr':\n this.removeHR();\n break;\n case 'lazy-linefeeds':\n this.convertLazyLinefeeds();\n break;\n case 'undo':\n this.undo();\n break;\n case 'redo':\n this.redo();\n break;\n }\n }\n \n /**\n * Copy content to clipboard\n */\n async copy(type) {\n const content = type === 'markdown' ? this._markdown : this._html;\n \n try {\n await navigator.clipboard.writeText(content);\n \n // Visual feedback\n const btn = this.toolbar.querySelector(`[data-action=\"copy-${type}\"]`);\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Copied!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n } catch (err) {\n console.error('Failed to copy:', err);\n }\n }\n \n // Public API\n \n /**\n * Get current markdown\n */\n get markdown() {\n return this._markdown;\n }\n \n /**\n * Set markdown content\n */\n set markdown(value) {\n this.setMarkdown(value);\n }\n \n /**\n * Get current HTML\n */\n get html() {\n return this._html;\n }\n \n /**\n * Get current mode\n */\n get mode() {\n return this.currentMode;\n }\n \n /**\n * Set markdown content\n */\n async setMarkdown(markdown) {\n // Wait for initialization if needed\n if (this.initPromise) {\n await this.initPromise;\n }\n \n this._markdown = markdown;\n if (this.sourceTextarea) {\n this.sourceTextarea.value = markdown;\n }\n this.updateFromMarkdown(markdown);\n }\n \n /**\n * Get markdown content\n */\n getMarkdown() {\n return this._markdown;\n }\n \n /**\n * Get HTML content\n */\n getHTML() {\n return this._html;\n }\n \n /**\n * Remove all horizontal rules (---) from markdown source.\n * Preserves content inside fences (``` or ~~~) and table separator rows.\n */\n async removeHR() {\n const cleaned = QuikdownEditor.removeHRFromMarkdown(this._markdown);\n await this.setMarkdown(cleaned);\n\n // Visual feedback if toolbar button exists\n const btn = this.toolbar?.querySelector('[data-action=\"remove-hr\"]');\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Removed!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n }\n\n /**\n * Static: remove horizontal rules from markdown string.\n * Safe for fences, tables, and all markdown constructs.\n * Can be used headless without an editor instance.\n * @param {string} markdown - source markdown\n * @returns {string} markdown with standalone HRs removed\n */\n static removeHRFromMarkdown(markdown) {\n const lines = (markdown || '').split('\\n');\n const result = [];\n let inFence = false;\n let fenceChar = null; // '`' or '~'\n let fenceLen = 0; // length of opening fence marker\n\n for (let i = 0; i < lines.length; i++) {\n const line = lines[i];\n const trimmed = line.trim();\n\n // Track fence open/close (``` or ~~~, 3+ chars)\n const fenceMatch = trimmed.match(/^(`{3,}|~{3,})/);\n if (fenceMatch) {\n const matchChar = fenceMatch[1][0];\n const matchLen = fenceMatch[1].length;\n if (!inFence) {\n inFence = true;\n fenceChar = matchChar;\n fenceLen = matchLen;\n result.push(line);\n continue;\n } else if (matchChar === fenceChar && matchLen >= fenceLen && /^(`{3,}|~{3,})\\s*$/.test(trimmed)) {\n // Closing fence: same char, at least as many chars, no trailing content\n inFence = false;\n fenceChar = null;\n fenceLen = 0;\n result.push(line);\n continue;\n }\n }\n\n // Inside a fence — keep everything\n if (inFence) {\n result.push(line);\n continue;\n }\n\n // Detect table row/separator with pipes — always keep\n if (/^\\|.*\\|$/.test(trimmed) || (/^[-| :]+$/.test(trimmed) && trimmed.includes('|'))) {\n result.push(line);\n continue;\n }\n\n // Check if this line is a standalone HR\n const isHR = /^[-_*](\\s*[-_*]){2,}\\s*$/.test(trimmed);\n if (isHR) {\n // Table separator heuristic: immediately adjacent lines (no blank\n // lines between) that look like table rows protect this HR-like line\n const prevLine = i > 0 ? lines[i - 1].trim() : '';\n const nextLine = i < lines.length - 1 ? lines[i + 1].trim() : '';\n if (_looksLikeTableRow(prevLine) || _looksLikeTableRow(nextLine)) {\n result.push(line);\n continue;\n }\n // It's a real HR — skip it\n continue;\n }\n\n result.push(line);\n }\n\n return result.join('\\n');\n }\n\n /**\n * Convert lazy linefeeds in markdown source.\n * Replaces single newlines with double newlines (adds real line breaks)\n * except inside fences, tables, and other block-level constructs.\n * Idempotent: calling multiple times produces the same result.\n * Can be used as a toolbar action or headless via the static method.\n */\n async convertLazyLinefeeds() {\n const converted = QuikdownEditor.convertLazyLinefeeds(this._markdown);\n await this.setMarkdown(converted);\n\n // Visual feedback if toolbar button exists\n const btn = this.toolbar?.querySelector('[data-action=\"lazy-linefeeds\"]');\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Converted!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n }\n\n /**\n * Static: convert lazy linefeeds in markdown source.\n * Turns single \\n between non-blank lines into \\n\\n so each line becomes\n * its own paragraph / hard break. Idempotent — already-doubled newlines\n * are not doubled again. Fences, tables, lists, blockquotes, headings,\n * and HTML blocks are left untouched.\n * @param {string} markdown - source markdown\n * @returns {string} markdown with lazy linefeeds resolved\n */\n static convertLazyLinefeeds(markdown) {\n // Two-phase approach (much cleaner than the old single pass):\n //\n // Phase A: walk lines, classify each as { content, blank, fence }.\n // Inside fences, lines are passed through verbatim.\n // Phase B: emit lines with the rule:\n // \"between two adjacent CONTENT lines, ensure exactly one\n // blank line — never zero, never more than one.\"\n //\n // The rule applies regardless of whether the content lines are\n // headings, lists, blockquotes, table rows, paragraphs, or HR — any\n // adjacent pair of non-fence non-blank lines gets exactly one blank\n // between them. This produces the cleanest possible output for any\n // input and is fully idempotent.\n //\n // Lines that are whitespace-only (e.g. \" \") are normalized to\n // empty strings, eliminating \"phantom\" blank lines.\n //\n // Lists are a special case: adjacent list items (same marker type)\n // should NOT get a blank line between them, otherwise we'd break\n // tight lists.\n //\n // Same applies to blockquote lines and table rows — adjacent rows\n // belong to the same block.\n\n const inputLines = (markdown || '').split('\\n');\n\n // -------- Phase A: classify lines, normalize whitespace-only --------\n // Each entry: { line, kind } where kind is one of:\n // 'fence-open', 'fence-close', 'fence-body', 'blank', 'content'\n // Plus a 'category' for content lines: 'list-ul', 'list-ol',\n // 'blockquote', 'table', 'heading', 'hr', 'paragraph'\n const items = [];\n let inFence = false;\n let fenceChar = null;\n let fenceLen = 0;\n\n for (const rawLine of inputLines) {\n const line = rawLine;\n const trimmed = line.trim();\n\n // Fence tracking\n const fenceMatch = trimmed.match(/^(`{3,}|~{3,})/);\n if (fenceMatch && !inFence) {\n inFence = true;\n fenceChar = fenceMatch[1][0];\n fenceLen = fenceMatch[1].length;\n items.push({ line, kind: 'fence-open' });\n continue;\n }\n if (inFence) {\n if (fenceMatch &&\n fenceMatch[1][0] === fenceChar &&\n fenceMatch[1].length >= fenceLen &&\n /^(`{3,}|~{3,})\\s*$/.test(trimmed)) {\n inFence = false;\n fenceChar = null;\n fenceLen = 0;\n items.push({ line, kind: 'fence-close' });\n } else {\n items.push({ line, kind: 'fence-body' });\n }\n continue;\n }\n\n // Outside fence: whitespace-only lines become canonical blanks\n if (trimmed === '') {\n items.push({ line: '', kind: 'blank' });\n continue;\n }\n\n // Categorize content lines so we can recognize adjacent same-kind blocks\n let category = 'paragraph';\n if (/^#{1,6}\\s/.test(trimmed)) category = 'heading';\n else if (/^[-_*](\\s*[-_*]){2,}\\s*$/.test(trimmed)) category = 'hr';\n else if (/^(\\d+\\.)\\s/.test(trimmed)) category = 'list-ol';\n else if (/^[-*+]\\s/.test(trimmed)) category = 'list-ul';\n else if (/^>/.test(trimmed)) category = 'blockquote';\n else if (/^\\|/.test(trimmed)) category = 'table';\n // Indented continuation of a list (2+ leading spaces or tab)\n else if (/^(?: {4}|\\t| {2,}[-*+]| {2,}\\d+\\.)/.test(line)) category = 'list-cont';\n\n items.push({ line, kind: 'content', category });\n }\n\n // -------- Phase B: emit with exactly-one-blank-line normalization --------\n // Same-block adjacent lines (lists, blockquotes, tables) stay\n // touching; any other adjacent content pair gets exactly one blank.\n const result = [];\n let prev = null; // last emitted non-blank content item\n\n function inSameBlock(a, b) {\n if (!a || !b) return false;\n // Lists: same marker family OR list-content continuation\n if ((a.category === 'list-ul' || a.category === 'list-ol' || a.category === 'list-cont') &&\n (b.category === 'list-ul' || b.category === 'list-ol' || b.category === 'list-cont')) {\n return true;\n }\n // Blockquotes\n if (a.category === 'blockquote' && b.category === 'blockquote') return true;\n // Table rows\n if (a.category === 'table' && b.category === 'table') return true;\n return false;\n }\n\n for (const item of items) {\n if (item.kind === 'fence-open' || item.kind === 'fence-body' || item.kind === 'fence-close') {\n // Fences: ensure exactly one blank line before the fence-open\n if (item.kind === 'fence-open' && prev && result.length > 0 && result[result.length - 1] !== '') {\n result.push('');\n }\n result.push(item.line);\n if (item.kind === 'fence-close') prev = { kind: 'content', category: 'fence' };\n continue;\n }\n\n if (item.kind === 'blank') {\n // Skip — Phase B inserts its own blank lines as needed\n continue;\n }\n\n // item.kind === 'content'\n if (prev) {\n if (inSameBlock(prev, item)) {\n // Adjacent same-block lines: no blank between\n } else {\n // Different blocks (or paragraphs): exactly one blank\n if (result[result.length - 1] !== '') result.push('');\n }\n }\n result.push(item.line);\n prev = item;\n }\n\n // Trim trailing blank lines so output has exactly one terminal newline\n while (result.length > 0 && result[result.length - 1] === '') result.pop();\n\n return result.join('\\n');\n }\n \n /**\n * Copy rendered content as rich text\n */\n async copyRendered() {\n try {\n const result = await getRenderedContent(this.previewPanel);\n if (result.success) {\n // Visual feedback\n const btn = this.toolbar?.querySelector('[data-action=\"copy-rendered\"]');\n if (btn) {\n const originalText = btn.textContent;\n btn.textContent = 'Copied!';\n setTimeout(() => {\n btn.textContent = originalText;\n }, 1500);\n }\n }\n } catch (err) {\n console.error('Failed to copy rendered content:', err);\n }\n }\n \n /**\n * Destroy the editor\n */\n destroy() {\n // Clear timers\n clearTimeout(this.updateTimer);\n \n // Clear container\n this.container.innerHTML = '';\n this.container.classList.remove('qde-container', 'qde-dark');\n \n // Remove injected styles (only if no other editors exist)\n const otherEditors = document.querySelectorAll('.qde-container');\n if (otherEditors.length === 0) {\n const style = document.getElementById('qde-styles');\n if (style) style.remove();\n }\n }\n}\n\n// --- Internal helpers for removeHR fence/table awareness ---\n\n/** Heuristic: does this line look like a markdown table row? */\nfunction _looksLikeTableRow(line) {\n return line.includes('|');\n}\n\n// Export\nexport default QuikdownEditor;\n\n// Export for CommonJS (needed for bundled ESM to work with Jest)\nif (typeof module !== 'undefined' && module.exports) {\n module.exports = QuikdownEditor;\n}\n\n// Also export for UMD builds\nif (typeof window !== 'undefined') {\n window.QuikdownEditor = QuikdownEditor;\n}"],"names":["CLASS_PREFIX","PLACEHOLDER_CB","ESC_MAP","QUIKDOWN_STYLES","h1","h2","h3","h4","h5","h6","pre","code","blockquote","table","th","td","hr","img","a","strong","em","del","ul","ol","li","quikdown","markdown","options","fence_plugin","inline_styles","bidirectional","lazy_linefeeds","getAttr","styles","tag","additionalStyle","style","includes","replace","trim","endsWith","classAttr","createGetAttr","escapeHtml","text","m","dataQd","marker","sanitizeUrl","url","allowUnsafe","trimmedUrl","lowerUrl","toLowerCase","dangerousProtocols","protocol","startsWith","html","codeBlocks","inlineCodes","match","fence","lang","placeholder","length","langTrimmed","render","push","trimEnd","custom","hasReverse","reverse","lines","split","result","inTable","tableLines","i","line","test","tableHtml","buildTable","join","processTable","hashes","content","level","listStack","indent","Math","floor","isOrdered","listType","listItemContent","taskListClass","taskMatch","checked","taskContent","isChecked","list","pop","type","currentList","liAttr","processLists","alt","src","sanitizedSrc","allow_unsafe_urls","altAttr","srcAttr","href","sanitizedHref","rel","textAttr","prefix","sanitizedUrl","forEach","pattern","blocks","bi","b","offset","substring","RegExp","replacement","block","undefined","langClass","codeAttr","langAttr","fenceAttr","processInlineMarkdown","separatorIndex","headerLines","slice","bodyLines","alignments","map","cell","trimmed","alignStyle","processedCell","quikdown_bd","async","svgToPng","svgElement","needsWhiteBackground","Promise","resolve","reject","svgString","XMLSerializer","serializeToString","canvas","document","createElement","ctx","getContext","Image","isMermaidSvg","closest","classList","contains","hasExplicitDimensions","getAttribute","svgWidth","svgHeight","clientWidth","viewBox","baseVal","width","parseFloat","clientHeight","height","modifiedSvgString","tempDiv","innerHTML","tempSvg","querySelector","setAttribute","toString","scale","onload","fillStyle","fillRect","drawImage","toBlob","blob","err","onerror","svgDataUrl","encodeURIComponent","rasterizeGeoJSONMap","liveContainer","_map","console","warn","mapRect","getBoundingClientRect","round","dpr","window","devicePixelRatio","tiles","querySelectorAll","tilePromises","tile","crossOrigin","tileRect","offsetX","left","offsetY","top","all","svgOverlays","svg","svgRect","svgStr","svgBlob","Blob","URL","createObjectURL","revokeObjectURL","Error","markerIcons","markerRect","toDataURL","error","getRenderedContent","previewPanel","mathBlocks","Array","from","some","MathJax","typesetPromise","clone","cloneNode","el","fontWeight","fontStyle","textDecoration","backgroundColor","padding","borderRadius","fontFamily","fontSize","marginTop","marginBottom","borderLeft","marginLeft","paddingLeft","color","border","borderTop","margin","borderCollapse","textAlign","parentElement","tr","lineHeight","whiteSpace","overflowX","appendChild","parentNode","replaceChild","images","naturalWidth","naturalHeight","maxWidth","maxHeight","min","random","substr","response","fetch","maxSize","size","dataUrl","reader","FileReader","onloadend","readAsDataURL","stlContainers","container","containerId","dataset","stlId","originalContainer","renderer","_threeRenderer","scene","_threeScene","camera","_threeCamera","imgWidth","imgHeight","canvasErr","cssText","textContent","mermaidContainers","pngBlob","chartContainers","chartId","svgContainers","mathElements","mathEl","value","_e","scaleFactor","scaledWidth","scaledHeight","scaleX","scaleY","imgElement","img2","setTimeout","displayWidth","displayHeight","geojsonContainers","clonedContainer","originalSource","allLiveContainers","candidate","mapContainers","id","leafletContainer","max","leafRect","r","x","y","w","h","overlaps","right","bottom","getComputedStyle","display","visibility","e","overlaySvgs","icon","mapDataUrl","String","htmlContainers","source","htmlImages","widthAttr","heightAttr","parseInt","tempImg","aspectRatio","absoluteImg","fragment","htmlContent","innerText","platform","navigator","userAgent","getPlatform","clipboard","write","ClipboardItem","success","modernErr","position","overflow","body","range","createRange","selectNodeContents","selection","getSelection","removeAllRanges","addRange","execCommand","removeChild","copyToClipboard","emitStyles","theme","themeOverrides","_textColor","css","Object","entries","themedStyle","oldColor","newColor","configure","version","module","exports","keys","key","toMarkdown","htmlOrElement","Element","walkNode","node","parentContext","nodeType","Node","TEXT_NODE","ELEMENT_NODE","tagName","childContent","child","childNodes","parentTag","repeat","boldMarker","emMarker","delMarker","codeMarker","fenceMarker","codeEl","quoteMarker","linkText","walkList","alignData","thead","headerRow","headers","_","align","tbody","row","cells","walkTable","trailingBlankLines","divLang","divFence","divSource","temp","mermaidPre","preSource","sourceElement","mermaidElement","listNode","depth","index","children","checkbox","itemContent","innerParser","DEFAULT_OPTIONS","mode","showToolbar","showRemoveHR","showLazyLinefeeds","debounceDelay","plugins","highlightjs","mermaid","preloadFences","customFences","enableComplexFences","showUndoRedo","undoStackSize","FENCE_LIBRARIES","check","hljs","script","cssDark","afterLoad","initialize","startOnLoad","math","beforeLoad","tex","inlineMath","displayMath","fontCache","startup","typeset","geojson","L","stl","THREE","QuikdownEditor","constructor","this","_markdown","_html","currentMode","updateTimer","_undoStack","_redoStack","_isUndoRedo","initPromise","init","loadPlugins","buildUI","attachEvents","applyTheme","setMode","initialContent","setMarkdown","add","toolbar","createToolbar","editorArea","className","sourcePanel","sourceTextarea","spellcheck","contentEditable","injectStyles","modeLabels","preview","btn","title","undoBtn","action","redoBtn","spacer","removeHRBtn","lazyLFBtn","getElementById","head","addEventListener","handleSourceInput","handlePreviewInput","target","handleAction","ctrlKey","metaKey","preventDefault","shiftKey","redo","undo","clearTimeout","updateFromMarkdown","updateFromHTML","_pushUndoState","createFencePlugin","makeFencesNonEditable","catch","_err","onChange","clonedPanel","preprocessSpecialElements","newMarkdown","_updateUndoButtons","panel","element","delimiter","csv","needsQuoting","renderSVG","renderHTML","renderMath","renderTable","renderJSON","renderMermaid","renderGeoJSON","renderSTL","getLanguage","highlight","language","doc","DOMParser","parseFromString","documentElement","remove","walker","createTreeWalker","NodeFilter","SHOW_ELEMENT","nextNode","attributes","attr","name","removeAttribute","outerHTML","errorContainer","message","Date","now","DOMPurify","clean","sanitize","lazyLoadLibrary","then","loaded","_lang","singleLineContent","ensureMathJaxLoaded","mathJaxLoading","loader","load","packages","processEscapes","processEnvironments","renderActions","addMenu","ignoreHtmlClass","processHtmlClass","escapedCode","header","parseCSVLine","current","inQuotes","char","nextChar","toHighlight","data","JSON","parse","stringify","mapId","renderMap","mapDiv","tileLayer","attribution","addTo","geoJsonLayer","geoJSON","getBounds","isValid","fitBounds","setView","_tileLayer","_geoJsonLayer","on","_qde_leaflet_loading","render3D","Scene","background","Color","PerspectiveCamera","WebGLRenderer","antialias","setSize","domElement","geometry","parseSTL","material","MeshLambertMaterial","mesh","Mesh","ambientLight","AmbientLight","directionalLight","DirectionalLight","set","normalize","box","Box3","setFromObject","center","getCenter","Vector3","getSize","maxDim","z","lookAt","animate","requestAnimationFrame","rotation","_qde_three_loading","stlData","BufferGeometry","vertices","normals","currentNormal","parts","Float32BufferAttribute","namesToLoad","Set","pf","n","isArray","entry","promises","lib","p","tasks","loadScript","loadCSS","_syncHljsTheme","scriptUrl","cssUrl","link","isDark","light","dark","disabled","_autoThemeListener","matchMedia","removeEventListener","mq","toggle","matches","setTheme","getTheme","setLazyLinefeeds","enabled","getLazyLinefeeds","setDebounceDelay","delay","getDebounceDelay","wasDark","onModeChange","splice","canUndo","previous","canRedo","next","clearHistory","copy","copyRendered","removeHR","convertLazyLinefeeds","writeText","originalText","getMarkdown","getHTML","cleaned","removeHRFromMarkdown","inFence","fenceChar","fenceLen","fenceMatch","matchChar","matchLen","prevLine","nextLine","_looksLikeTableRow","converted","inputLines","items","rawLine","kind","category","prev","inSameBlock","item","destroy"],"mappings":"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"}
|