@intlayer/core 8.3.0-canary.2 → 8.3.0-canary.4
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.
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`./constants.cjs`),t=e=>{let t=e.length;for(;t>0&&e[t-1]<=` `;)t--;return e.slice(0,t)},n=(e,t)=>e.startsWith(t),r=e=>{let t=e[0];return(t===`"`||t===`'`)&&e.length>=2&&e[e.length-1]===t?e.slice(1,-1):e},i=t=>t&&t.replace(e.UNESCAPE_R,`$1`),a=(...e)=>e.filter(Boolean).join(` `),o=(e,t,n)=>{let r=e,i=t.split(`.`);for(;i.length&&(r=r[i[0]],r!==void 0);)i.shift();return r??n},s=e=>e.replace(/[ÀÁÂÃÄÅàáâãä忯]/g,`a`).replace(/[çÇ]/g,`c`).replace(/[ðÐ]/g,`d`).replace(/[ÈÉÊËéèêë]/g,`e`).replace(/[ÏïÎîÍíÌì]/g,`i`).replace(/[Ññ]/g,`n`).replace(/[øØœŒÕõÔôÓóÒò]/g,`o`).replace(/[ÜüÛûÚúÙù]/g,`u`).replace(/[ŸÿÝý]/g,`y`).replace(/[^a-z0-9- ]/gi,``).replace(/ /gi,`-`).toLowerCase(),c=/(javascript|vbscript|data(?!:image)):/i,l=e=>{try{let t=decodeURIComponent(e).replace(/[^A-Za-z0-9/:]/g,``);if(c.test(t))return
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});const e=require(`./constants.cjs`),t=e=>{let t=e.length;for(;t>0&&e[t-1]<=` `;)t--;return e.slice(0,t)},n=(e,t)=>e.startsWith(t),r=e=>{let t=e[0];return(t===`"`||t===`'`)&&e.length>=2&&e[e.length-1]===t?e.slice(1,-1):e},i=t=>t&&t.replace(e.UNESCAPE_R,`$1`),a=(...e)=>e.filter(Boolean).join(` `),o=(e,t,n)=>{let r=e,i=t.split(`.`);for(;i.length&&(r=r[i[0]],r!==void 0);)i.shift();return r??n},s=e=>e.replace(/[ÀÁÂÃÄÅàáâãä忯]/g,`a`).replace(/[çÇ]/g,`c`).replace(/[ðÐ]/g,`d`).replace(/[ÈÉÊËéèêë]/g,`e`).replace(/[ÏïÎîÍíÌì]/g,`i`).replace(/[Ññ]/g,`n`).replace(/[øØœŒÕõÔôÓóÒò]/g,`o`).replace(/[ÜüÛûÚúÙù]/g,`u`).replace(/[ŸÿÝý]/g,`y`).replace(/[^a-z0-9- ]/gi,``).replace(/ /gi,`-`).toLowerCase(),c=/(javascript|vbscript|data(?!:image)):/i,l=e=>{try{let t=decodeURIComponent(e).replace(/[^A-Za-z0-9/:]/g,``);if(c.test(t))return console.warn(`Input contains an unsafe JavaScript/VBScript/data expression, it will not be rendered.`,t),null}catch{return console.warn(`Input could not be decoded due to malformed syntax or characters, it will not be rendered.`,e),null}return e},u=t=>{let n=performance.now(),r=t.replace(e.CR_NEWLINE_R,`
|
|
2
2
|
`).replace(e.FORMFEED_R,``).replace(e.TAB_R,` `),i=performance.now()-n;return i>20&&console.log(`normalizeWhitespace: ${i.toFixed(3)}ms, source length: ${t.length}`),r},d=(e,t)=>{let n=performance.now();if(!t)return e;let r=e.split(`
|
|
3
3
|
`),i=!1,a=null,o=e=>e.match(/^\s*(`{3,}|~{3,})/),s=e=>{let t=o(e);if(!t)return;let n=t[1];i?a&&e.includes(a)&&(i=!1,a=null):(i=!0,a=n)},c=r.map(e=>{if(o(e)){let n=e.startsWith(t)?e.slice(t.length):e;return s(e),n}return i?e:e.startsWith(t)?e.slice(t.length):e}).join(`
|
|
4
4
|
`),l=performance.now()-n;return l>20&&console.log(`trimLeadingWhitespaceOutsideFences: ${l.toFixed(3)}ms, text length: ${e.length}, lines count: ${r.length}`),c},f=t=>(t.indexOf(`-`)!==-1&&t.match(e.HTML_CUSTOM_ATTR_R)===null&&(t=t.replace(e.CAPTURE_LETTER_AFTER_HYPHEN,(e,t)=>t.toUpperCase())),t),p=e=>{let t=performance.now(),n=[],r=``,i=!1,a=!1,o=``;if(!e)return n;for(let t=0;t<e.length;t++){let s=e[t];if((s===`"`||s===`'`)&&!i&&(a?s===o&&(a=!1,o=``):(a=!0,o=s)),s===`(`&&r.endsWith(`url`)?i=!0:s===`)`&&i&&(i=!1),s===`;`&&!a&&!i){let e=r.trim();if(e){let t=e.indexOf(`:`);if(t>0){let r=e.slice(0,t).trim(),i=e.slice(t+1).trim();n.push([r,i])}}r=``}else r+=s}let s=r.trim();if(s){let e=s.indexOf(`:`);if(e>0){let t=s.slice(0,e).trim(),r=s.slice(e+1).trim();n.push([t,r])}}let c=performance.now()-t;return c>20&&console.log(`parseStyleAttribute: ${c.toFixed(3)}ms, styleString length: ${e.length}, styles count: ${n.length}`),n},m=(t,n,r,a)=>n===`style`?p(r).reduce((e,[n,r])=>{let i=n.replace(/(-[a-z])/g,e=>e[1].toUpperCase());return e[i]=a(r,t,n),e},{}):e.ATTRIBUTES_TO_SANITIZE.indexOf(n)===-1?(r.match(e.INTERPOLATION_R)&&(r=i(r.slice(1,r.length-1))),r===`true`?!0:r===`false`?!1:r):a(i(r),t,n),h=t=>e.TABLE_RIGHT_ALIGN.test(t)?`right`:e.TABLE_CENTER_ALIGN.test(t)?`center`:(e.TABLE_LEFT_ALIGN.test(t),`left`),g=t=>t.replace(e.TABLE_TRIM_PIPES,``).split(`|`).map(h),_=(e,t,n,r)=>{let i=performance.now(),a=n.inTable;n.inTable=!0;let o=[[]],s=``,c=()=>{if(!s)return;let e=o[o.length-1];e.push.apply(e,t(s,n)),s=``};e.trim().split(/(`[^`]*`|\\\||\|)/).filter(Boolean).forEach((e,t,n)=>{if(e.trim()===`|`&&(c(),r)){t!==0&&t!==n.length-1&&o.push([]);return}s+=e}),c(),n.inTable=a;let l=performance.now()-i;return l>20&&console.log(`parseTableRow: ${l.toFixed(3)}ms, source length: ${e.length}, cells count: ${o.length}`),o},v=(e,t,n)=>{let r=performance.now(),i=e.trim().split(`
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.cjs","names":["UNESCAPE_R","CR_NEWLINE_R","FORMFEED_R","TAB_R","HTML_CUSTOM_ATTR_R","CAPTURE_LETTER_AFTER_HYPHEN","ATTRIBUTES_TO_SANITIZE","INTERPOLATION_R","TABLE_RIGHT_ALIGN","TABLE_CENTER_ALIGN","TABLE_LEFT_ALIGN","TABLE_TRIM_PIPES"],"sources":["../../../src/markdown/utils.ts"],"sourcesContent":["import {\n ATTRIBUTES_TO_SANITIZE,\n CAPTURE_LETTER_AFTER_HYPHEN,\n CR_NEWLINE_R,\n DURATION_DELAY_TRIGGER,\n FORMFEED_R,\n HTML_CUSTOM_ATTR_R,\n INTERPOLATION_R,\n TAB_R,\n TABLE_CENTER_ALIGN,\n TABLE_LEFT_ALIGN,\n TABLE_RIGHT_ALIGN,\n TABLE_TRIM_PIPES,\n UNESCAPE_R,\n} from './constants';\nimport type { NestedParser, ParserResult, ParseState, Rule } from './types';\n\n// ============================================================================\n// STRING UTILITIES\n// ============================================================================\n\n/**\n * Trim trailing whitespace from a string.\n */\nexport const trimEnd = (str: string): string => {\n let end = str.length;\n\n while (end > 0 && str[end - 1] <= ' ') end--;\n\n return str.slice(0, end);\n};\n\n/**\n * Check if string starts with prefix.\n */\nexport const startsWith = (str: string, prefix: string): boolean => {\n return str.startsWith(prefix);\n};\n\n/**\n * Remove symmetrical leading and trailing quotes.\n */\nexport const unquote = (str: string): string => {\n const first = str[0];\n\n if (\n (first === '\"' || first === \"'\") &&\n str.length >= 2 &&\n str[str.length - 1] === first\n ) {\n return str.slice(1, -1);\n }\n\n return str;\n};\n\n/**\n * Unescape backslash-escaped characters.\n */\nexport const unescapeString = (rawString: string): string =>\n rawString ? rawString.replace(UNESCAPE_R, '$1') : rawString;\n\n/**\n * Join class names, filtering out falsy values.\n */\nexport const cx = (...args: any[]): string => args.filter(Boolean).join(' ');\n\n/**\n * Get a nested property from an object using dot notation.\n */\nexport const get = (src: any, path: string, fb?: any): any => {\n let ptr = src;\n const frags = path.split('.');\n\n while (frags.length) {\n ptr = ptr[frags[0]];\n\n if (ptr === undefined) break;\n else frags.shift();\n }\n\n return ptr ?? fb;\n};\n\n// ============================================================================\n// SLUGIFY\n// ============================================================================\n\n/**\n * Convert a string to a URL-safe slug.\n * Based on https://stackoverflow.com/a/18123682/1141611\n */\nexport const slugify = (str: string): string =>\n str\n .replace(/[ÀÁÂÃÄÅàáâãä忯]/g, 'a')\n .replace(/[çÇ]/g, 'c')\n .replace(/[ðÐ]/g, 'd')\n .replace(/[ÈÉÊËéèêë]/g, 'e')\n .replace(/[ÏïÎîÍíÌì]/g, 'i')\n .replace(/[Ññ]/g, 'n')\n .replace(/[øØœŒÕõÔôÓóÒò]/g, 'o')\n .replace(/[ÜüÛûÚúÙù]/g, 'u')\n .replace(/[ŸÿÝý]/g, 'y')\n .replace(/[^a-z0-9- ]/gi, '')\n .replace(/ /gi, '-')\n .toLowerCase();\n\n// ============================================================================\n// SANITIZER\n// ============================================================================\n\nconst SANITIZE_R = /(javascript|vbscript|data(?!:image)):/i;\n\n/**\n * Sanitize URLs to prevent XSS attacks.\n * Returns null if the URL is unsafe.\n */\nexport const sanitizer = (input: string): string | null => {\n try {\n const decoded = decodeURIComponent(input).replace(/[^A-Za-z0-9/:]/g, '');\n\n if (SANITIZE_R.test(decoded)) {\n if (process.env.NODE_ENV !== 'production') {\n console.warn(\n 'Input contains an unsafe JavaScript/VBScript/data expression, it will not be rendered.',\n decoded\n );\n }\n\n return null;\n }\n } catch (_e) {\n if (process.env.NODE_ENV !== 'production') {\n console.warn(\n 'Input could not be decoded due to malformed syntax or characters, it will not be rendered.',\n input\n );\n }\n\n // decodeURIComponent sometimes throws a URIError\n return null;\n }\n\n return input;\n};\n\n// ============================================================================\n// WHITESPACE NORMALIZATION\n// ============================================================================\n\n/**\n * Normalize whitespace in source string.\n */\nexport const normalizeWhitespace = (source: string): string => {\n const start = performance.now();\n const result = source\n .replace(CR_NEWLINE_R, '\\n')\n .replace(FORMFEED_R, '')\n .replace(TAB_R, ' ');\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `normalizeWhitespace: ${duration.toFixed(3)}ms, source length: ${source.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Safely remove a uniform leading indentation from lines, but do NOT touch\n * the content inside fenced code blocks (``` or ~~~).\n */\nexport const trimLeadingWhitespaceOutsideFences = (\n text: string,\n whitespace: string\n): string => {\n const start = performance.now();\n if (!whitespace) return text;\n\n const lines = text.split('\\n');\n let inFence = false;\n let fenceToken: string | null = null;\n\n const isFenceLine = (line: string): RegExpMatchArray | null =>\n line.match(/^\\s*(`{3,}|~{3,})/);\n\n const maybeToggleFence = (line: string): void => {\n const m = isFenceLine(line);\n\n if (!m) return;\n\n const token = m[1];\n\n if (!inFence) {\n inFence = true;\n fenceToken = token;\n } else if (fenceToken && line.includes(fenceToken)) {\n inFence = false;\n fenceToken = null;\n }\n };\n\n const out = lines.map((line) => {\n const fenceMatch = isFenceLine(line);\n if (fenceMatch) {\n const trimmedFenceLine = line.startsWith(whitespace)\n ? line.slice(whitespace.length)\n : line;\n maybeToggleFence(line);\n return trimmedFenceLine;\n }\n\n if (inFence) {\n return line;\n }\n\n return line.startsWith(whitespace) ? line.slice(whitespace.length) : line;\n });\n\n const result = out.join('\\n');\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `trimLeadingWhitespaceOutsideFences: ${duration.toFixed(3)}ms, text length: ${text.length}, lines count: ${lines.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Normalize HTML attribute key to JSX prop name.\n */\nexport const normalizeAttributeKey = (key: string): string => {\n const hyphenIndex = key.indexOf('-');\n\n if (hyphenIndex !== -1 && key.match(HTML_CUSTOM_ATTR_R) === null) {\n key = key.replace(CAPTURE_LETTER_AFTER_HYPHEN, (_, letter) => {\n return letter.toUpperCase();\n });\n }\n\n return key;\n};\n\ntype StyleTuple = [key: string, value: string];\n\n/**\n * Parse a CSS style string into an array of [key, value] tuples.\n */\nexport const parseStyleAttribute = (styleString: string): StyleTuple[] => {\n const start = performance.now();\n const styles: StyleTuple[] = [];\n let buffer = '';\n let inUrl = false;\n let inQuotes = false;\n let quoteChar: '\"' | \"'\" | '' = '';\n\n if (!styleString) return styles;\n\n for (let i = 0; i < styleString.length; i++) {\n const char = styleString[i];\n\n if ((char === '\"' || char === \"'\") && !inUrl) {\n if (!inQuotes) {\n inQuotes = true;\n quoteChar = char;\n } else if (char === quoteChar) {\n inQuotes = false;\n quoteChar = '';\n }\n }\n\n if (char === '(' && buffer.endsWith('url')) {\n inUrl = true;\n } else if (char === ')' && inUrl) {\n inUrl = false;\n }\n\n if (char === ';' && !inQuotes && !inUrl) {\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n buffer = '';\n } else {\n buffer += char;\n }\n }\n\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseStyleAttribute: ${duration.toFixed(3)}ms, styleString length: ${styleString.length}, styles count: ${styles.length}`\n );\n }\n\n return styles;\n};\n\n/**\n * Convert an attribute value to a Node prop value.\n */\nexport const attributeValueToNodePropValue = (\n tag: string,\n key: string,\n value: string,\n sanitizeUrlFn: (\n value: string,\n tag: string,\n attribute: string\n ) => string | null\n): any => {\n if (key === 'style') {\n return parseStyleAttribute(value).reduce(\n (styles, [styleKey, styleValue]) => {\n const camelCasedKey = styleKey.replace(/(-[a-z])/g, (substr) =>\n substr[1].toUpperCase()\n );\n\n (styles as Record<string, any>)[camelCasedKey] = sanitizeUrlFn(\n styleValue,\n tag,\n styleKey\n );\n\n return styles;\n },\n {} as Record<string, any>\n );\n } else if (ATTRIBUTES_TO_SANITIZE.indexOf(key) !== -1) {\n return sanitizeUrlFn(unescapeString(value), tag, key);\n } else if (value.match(INTERPOLATION_R)) {\n value = unescapeString(value.slice(1, value.length - 1));\n }\n\n if (value === 'true') {\n return true;\n } else if (value === 'false') {\n return false;\n }\n\n return value;\n};\n\n// ============================================================================\n// TABLE PARSING\n// ============================================================================\n\n/**\n * Parse table alignment from a separator row.\n */\nexport const parseTableAlignCapture = (\n alignCapture: string\n): 'left' | 'right' | 'center' => {\n if (TABLE_RIGHT_ALIGN.test(alignCapture)) {\n return 'right';\n } else if (TABLE_CENTER_ALIGN.test(alignCapture)) {\n return 'center';\n } else if (TABLE_LEFT_ALIGN.test(alignCapture)) {\n return 'left';\n }\n\n return 'left';\n};\n\n/**\n * Parse table alignment row.\n */\nexport const parseTableAlign = (\n source: string\n): ('left' | 'right' | 'center')[] => {\n const alignText = source.replace(TABLE_TRIM_PIPES, '').split('|');\n return alignText.map(parseTableAlignCapture);\n};\n\n/**\n * Parse a single table row.\n */\nexport const parseTableRow = (\n source: string,\n parse: NestedParser,\n state: ParseState,\n tableOutput: boolean\n): ParserResult[][] => {\n const start = performance.now();\n const prevInTable = state.inTable;\n\n state.inTable = true;\n\n const cells: ParserResult[][] = [[]];\n let acc = '';\n\n const flush = (): void => {\n if (!acc) return;\n\n const cell = cells[cells.length - 1];\n cell.push.apply(cell, parse(acc, state));\n acc = '';\n };\n\n source\n .trim()\n .split(/(`[^`]*`|\\\\\\||\\|)/)\n .filter(Boolean)\n .forEach((fragment, i, arr) => {\n if (fragment.trim() === '|') {\n flush();\n\n if (tableOutput) {\n if (i !== 0 && i !== arr.length - 1) {\n cells.push([]);\n }\n\n return;\n }\n }\n\n acc += fragment;\n });\n\n flush();\n\n state.inTable = prevInTable;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableRow: ${duration.toFixed(3)}ms, source length: ${source.length}, cells count: ${cells.length}`\n );\n }\n\n return cells;\n};\n\n/**\n * Parse table cells (multiple rows).\n */\nexport const parseTableCells = (\n source: string,\n parse: NestedParser,\n state: ParseState\n): ParserResult[][][] => {\n const start = performance.now();\n const rowsText = source.trim().split('\\n');\n\n const result = rowsText.map((rowText) =>\n parseTableRow(rowText, parse, state, true)\n );\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableCells: ${duration.toFixed(3)}ms, source length: ${source.length}, rows count: ${rowsText.length}`\n );\n }\n\n return result;\n};\n\n// ============================================================================\n// PARSING HELPERS\n// ============================================================================\n\n/**\n * Check if a rule qualifies for the current source and state.\n */\nexport const qualifies = (\n source: string,\n state: ParseState,\n qualify: NonNullable<Rule<any>['_qualify']>\n): boolean => {\n if (Array.isArray(qualify)) {\n for (let i = 0; i < qualify.length; i++) {\n if (startsWith(source, qualify[i])) return true;\n }\n\n return false;\n }\n\n return (qualify as (source: string, state: ParseState) => boolean)(\n source,\n state\n );\n};\n\n/**\n * Marks a matcher function as eligible for being run inside an inline context.\n */\nexport const allowInline = <T extends (...args: any[]) => any>(\n fn: T\n): T & { inline: 1 } => {\n (fn as any).inline = 1;\n return fn as T & { inline: 1 };\n};\n\n/**\n * Creates a match function for an inline scoped element from a regex.\n */\nexport const inlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for inline elements except links.\n */\nexport const simpleInlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for a block scoped element from a regex.\n */\nexport const blockRegex =\n (regex: RegExp) =>\n (source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return null;\n } else {\n return regex.exec(source);\n }\n };\n\n/**\n * Creates a match function from a regex, ignoring block/inline scope.\n */\nexport const anyScopeRegex = (\n fn: RegExp | ((source: string, state: ParseState) => RegExpMatchArray | null)\n) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (typeof fn === 'function') {\n return fn(source, state);\n }\n return fn.exec(source);\n });\n\n/**\n * Parse inline content (including links).\n */\nexport const parseInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n state.inline = true;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse simple inline content (no links).\n */\nexport const parseSimpleInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n\n state.inline = false;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseSimpleInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse block content.\n */\nexport const parseBlock = (\n parse: NestedParser,\n children: string,\n state: ParseState = {}\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline || false;\n state.inline = false;\n const normalizedChildren = trimEnd(children);\n const needsTerminator = /\\n\\n$/.test(normalizedChildren) === false;\n const blockInput = needsTerminator\n ? normalizedChildren.endsWith('\\n')\n ? `${normalizedChildren}\\n`\n : `${normalizedChildren}\\n\\n`\n : normalizedChildren;\n\n const result = parse(blockInput, state);\n state.inline = isCurrentlyInline;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseBlock: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Helper to parse capture group 2 as inline content.\n */\nexport const parseCaptureInline = (\n capture: RegExpMatchArray,\n parse: NestedParser,\n state: ParseState\n): { children: ParserResult[] } => {\n return {\n children: parseInline(parse, capture[2], state),\n };\n};\n\n/**\n * Helper that captures nothing (empty object).\n */\nexport const captureNothing = (): Record<string, never> => ({});\n\n/**\n * Helper that renders nothing (null).\n */\nexport const renderNothing = (): null => null;\n\n/**\n * Check if any regex in a list matches the input.\n */\nexport const some = (regexes: RegExp[], input: string): boolean => {\n for (let i = 0; i < regexes.length; i++) {\n if (regexes[i].test(input)) {\n return true;\n }\n }\n return false;\n};\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"utils.cjs","names":["UNESCAPE_R","CR_NEWLINE_R","FORMFEED_R","TAB_R","HTML_CUSTOM_ATTR_R","CAPTURE_LETTER_AFTER_HYPHEN","ATTRIBUTES_TO_SANITIZE","INTERPOLATION_R","TABLE_RIGHT_ALIGN","TABLE_CENTER_ALIGN","TABLE_LEFT_ALIGN","TABLE_TRIM_PIPES"],"sources":["../../../src/markdown/utils.ts"],"sourcesContent":["import {\n ATTRIBUTES_TO_SANITIZE,\n CAPTURE_LETTER_AFTER_HYPHEN,\n CR_NEWLINE_R,\n DURATION_DELAY_TRIGGER,\n FORMFEED_R,\n HTML_CUSTOM_ATTR_R,\n INTERPOLATION_R,\n TAB_R,\n TABLE_CENTER_ALIGN,\n TABLE_LEFT_ALIGN,\n TABLE_RIGHT_ALIGN,\n TABLE_TRIM_PIPES,\n UNESCAPE_R,\n} from './constants';\nimport type { NestedParser, ParserResult, ParseState, Rule } from './types';\n\n// ============================================================================\n// STRING UTILITIES\n// ============================================================================\n\n/**\n * Trim trailing whitespace from a string.\n */\nexport const trimEnd = (str: string): string => {\n let end = str.length;\n\n while (end > 0 && str[end - 1] <= ' ') end--;\n\n return str.slice(0, end);\n};\n\n/**\n * Check if string starts with prefix.\n */\nexport const startsWith = (str: string, prefix: string): boolean => {\n return str.startsWith(prefix);\n};\n\n/**\n * Remove symmetrical leading and trailing quotes.\n */\nexport const unquote = (str: string): string => {\n const first = str[0];\n\n if (\n (first === '\"' || first === \"'\") &&\n str.length >= 2 &&\n str[str.length - 1] === first\n ) {\n return str.slice(1, -1);\n }\n\n return str;\n};\n\n/**\n * Unescape backslash-escaped characters.\n */\nexport const unescapeString = (rawString: string): string =>\n rawString ? rawString.replace(UNESCAPE_R, '$1') : rawString;\n\n/**\n * Join class names, filtering out falsy values.\n */\nexport const cx = (...args: any[]): string => args.filter(Boolean).join(' ');\n\n/**\n * Get a nested property from an object using dot notation.\n */\nexport const get = (src: any, path: string, fb?: any): any => {\n let ptr = src;\n const frags = path.split('.');\n\n while (frags.length) {\n ptr = ptr[frags[0]];\n\n if (ptr === undefined) break;\n else frags.shift();\n }\n\n return ptr ?? fb;\n};\n\n// ============================================================================\n// SLUGIFY\n// ============================================================================\n\n/**\n * Convert a string to a URL-safe slug.\n * Based on https://stackoverflow.com/a/18123682/1141611\n */\nexport const slugify = (str: string): string =>\n str\n .replace(/[ÀÁÂÃÄÅàáâãä忯]/g, 'a')\n .replace(/[çÇ]/g, 'c')\n .replace(/[ðÐ]/g, 'd')\n .replace(/[ÈÉÊËéèêë]/g, 'e')\n .replace(/[ÏïÎîÍíÌì]/g, 'i')\n .replace(/[Ññ]/g, 'n')\n .replace(/[øØœŒÕõÔôÓóÒò]/g, 'o')\n .replace(/[ÜüÛûÚúÙù]/g, 'u')\n .replace(/[ŸÿÝý]/g, 'y')\n .replace(/[^a-z0-9- ]/gi, '')\n .replace(/ /gi, '-')\n .toLowerCase();\n\n// ============================================================================\n// SANITIZER\n// ============================================================================\n\nconst SANITIZE_R = /(javascript|vbscript|data(?!:image)):/i;\n\n/**\n * Sanitize URLs to prevent XSS attacks.\n * Returns null if the URL is unsafe.\n */\nexport const sanitizer = (input: string): string | null => {\n try {\n const decoded = decodeURIComponent(input).replace(/[^A-Za-z0-9/:]/g, '');\n\n if (SANITIZE_R.test(decoded)) {\n console.warn(\n 'Input contains an unsafe JavaScript/VBScript/data expression, it will not be rendered.',\n decoded\n );\n\n return null;\n }\n } catch (_e) {\n console.warn(\n 'Input could not be decoded due to malformed syntax or characters, it will not be rendered.',\n input\n );\n\n // decodeURIComponent sometimes throws a URIError\n return null;\n }\n\n return input;\n};\n\n// ============================================================================\n// WHITESPACE NORMALIZATION\n// ============================================================================\n\n/**\n * Normalize whitespace in source string.\n */\nexport const normalizeWhitespace = (source: string): string => {\n const start = performance.now();\n const result = source\n .replace(CR_NEWLINE_R, '\\n')\n .replace(FORMFEED_R, '')\n .replace(TAB_R, ' ');\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `normalizeWhitespace: ${duration.toFixed(3)}ms, source length: ${source.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Safely remove a uniform leading indentation from lines, but do NOT touch\n * the content inside fenced code blocks (``` or ~~~).\n */\nexport const trimLeadingWhitespaceOutsideFences = (\n text: string,\n whitespace: string\n): string => {\n const start = performance.now();\n if (!whitespace) return text;\n\n const lines = text.split('\\n');\n let inFence = false;\n let fenceToken: string | null = null;\n\n const isFenceLine = (line: string): RegExpMatchArray | null =>\n line.match(/^\\s*(`{3,}|~{3,})/);\n\n const maybeToggleFence = (line: string): void => {\n const m = isFenceLine(line);\n\n if (!m) return;\n\n const token = m[1];\n\n if (!inFence) {\n inFence = true;\n fenceToken = token;\n } else if (fenceToken && line.includes(fenceToken)) {\n inFence = false;\n fenceToken = null;\n }\n };\n\n const out = lines.map((line) => {\n const fenceMatch = isFenceLine(line);\n if (fenceMatch) {\n const trimmedFenceLine = line.startsWith(whitespace)\n ? line.slice(whitespace.length)\n : line;\n maybeToggleFence(line);\n return trimmedFenceLine;\n }\n\n if (inFence) {\n return line;\n }\n\n return line.startsWith(whitespace) ? line.slice(whitespace.length) : line;\n });\n\n const result = out.join('\\n');\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `trimLeadingWhitespaceOutsideFences: ${duration.toFixed(3)}ms, text length: ${text.length}, lines count: ${lines.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Normalize HTML attribute key to JSX prop name.\n */\nexport const normalizeAttributeKey = (key: string): string => {\n const hyphenIndex = key.indexOf('-');\n\n if (hyphenIndex !== -1 && key.match(HTML_CUSTOM_ATTR_R) === null) {\n key = key.replace(CAPTURE_LETTER_AFTER_HYPHEN, (_, letter) => {\n return letter.toUpperCase();\n });\n }\n\n return key;\n};\n\ntype StyleTuple = [key: string, value: string];\n\n/**\n * Parse a CSS style string into an array of [key, value] tuples.\n */\nexport const parseStyleAttribute = (styleString: string): StyleTuple[] => {\n const start = performance.now();\n const styles: StyleTuple[] = [];\n let buffer = '';\n let inUrl = false;\n let inQuotes = false;\n let quoteChar: '\"' | \"'\" | '' = '';\n\n if (!styleString) return styles;\n\n for (let i = 0; i < styleString.length; i++) {\n const char = styleString[i];\n\n if ((char === '\"' || char === \"'\") && !inUrl) {\n if (!inQuotes) {\n inQuotes = true;\n quoteChar = char;\n } else if (char === quoteChar) {\n inQuotes = false;\n quoteChar = '';\n }\n }\n\n if (char === '(' && buffer.endsWith('url')) {\n inUrl = true;\n } else if (char === ')' && inUrl) {\n inUrl = false;\n }\n\n if (char === ';' && !inQuotes && !inUrl) {\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n buffer = '';\n } else {\n buffer += char;\n }\n }\n\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseStyleAttribute: ${duration.toFixed(3)}ms, styleString length: ${styleString.length}, styles count: ${styles.length}`\n );\n }\n\n return styles;\n};\n\n/**\n * Convert an attribute value to a Node prop value.\n */\nexport const attributeValueToNodePropValue = (\n tag: string,\n key: string,\n value: string,\n sanitizeUrlFn: (\n value: string,\n tag: string,\n attribute: string\n ) => string | null\n): any => {\n if (key === 'style') {\n return parseStyleAttribute(value).reduce(\n (styles, [styleKey, styleValue]) => {\n const camelCasedKey = styleKey.replace(/(-[a-z])/g, (substr) =>\n substr[1].toUpperCase()\n );\n\n (styles as Record<string, any>)[camelCasedKey] = sanitizeUrlFn(\n styleValue,\n tag,\n styleKey\n );\n\n return styles;\n },\n {} as Record<string, any>\n );\n } else if (ATTRIBUTES_TO_SANITIZE.indexOf(key) !== -1) {\n return sanitizeUrlFn(unescapeString(value), tag, key);\n } else if (value.match(INTERPOLATION_R)) {\n value = unescapeString(value.slice(1, value.length - 1));\n }\n\n if (value === 'true') {\n return true;\n } else if (value === 'false') {\n return false;\n }\n\n return value;\n};\n\n// ============================================================================\n// TABLE PARSING\n// ============================================================================\n\n/**\n * Parse table alignment from a separator row.\n */\nexport const parseTableAlignCapture = (\n alignCapture: string\n): 'left' | 'right' | 'center' => {\n if (TABLE_RIGHT_ALIGN.test(alignCapture)) {\n return 'right';\n } else if (TABLE_CENTER_ALIGN.test(alignCapture)) {\n return 'center';\n } else if (TABLE_LEFT_ALIGN.test(alignCapture)) {\n return 'left';\n }\n\n return 'left';\n};\n\n/**\n * Parse table alignment row.\n */\nexport const parseTableAlign = (\n source: string\n): ('left' | 'right' | 'center')[] => {\n const alignText = source.replace(TABLE_TRIM_PIPES, '').split('|');\n return alignText.map(parseTableAlignCapture);\n};\n\n/**\n * Parse a single table row.\n */\nexport const parseTableRow = (\n source: string,\n parse: NestedParser,\n state: ParseState,\n tableOutput: boolean\n): ParserResult[][] => {\n const start = performance.now();\n const prevInTable = state.inTable;\n\n state.inTable = true;\n\n const cells: ParserResult[][] = [[]];\n let acc = '';\n\n const flush = (): void => {\n if (!acc) return;\n\n const cell = cells[cells.length - 1];\n cell.push.apply(cell, parse(acc, state));\n acc = '';\n };\n\n source\n .trim()\n .split(/(`[^`]*`|\\\\\\||\\|)/)\n .filter(Boolean)\n .forEach((fragment, i, arr) => {\n if (fragment.trim() === '|') {\n flush();\n\n if (tableOutput) {\n if (i !== 0 && i !== arr.length - 1) {\n cells.push([]);\n }\n\n return;\n }\n }\n\n acc += fragment;\n });\n\n flush();\n\n state.inTable = prevInTable;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableRow: ${duration.toFixed(3)}ms, source length: ${source.length}, cells count: ${cells.length}`\n );\n }\n\n return cells;\n};\n\n/**\n * Parse table cells (multiple rows).\n */\nexport const parseTableCells = (\n source: string,\n parse: NestedParser,\n state: ParseState\n): ParserResult[][][] => {\n const start = performance.now();\n const rowsText = source.trim().split('\\n');\n\n const result = rowsText.map((rowText) =>\n parseTableRow(rowText, parse, state, true)\n );\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableCells: ${duration.toFixed(3)}ms, source length: ${source.length}, rows count: ${rowsText.length}`\n );\n }\n\n return result;\n};\n\n// ============================================================================\n// PARSING HELPERS\n// ============================================================================\n\n/**\n * Check if a rule qualifies for the current source and state.\n */\nexport const qualifies = (\n source: string,\n state: ParseState,\n qualify: NonNullable<Rule<any>['_qualify']>\n): boolean => {\n if (Array.isArray(qualify)) {\n for (let i = 0; i < qualify.length; i++) {\n if (startsWith(source, qualify[i])) return true;\n }\n\n return false;\n }\n\n return (qualify as (source: string, state: ParseState) => boolean)(\n source,\n state\n );\n};\n\n/**\n * Marks a matcher function as eligible for being run inside an inline context.\n */\nexport const allowInline = <T extends (...args: any[]) => any>(\n fn: T\n): T & { inline: 1 } => {\n (fn as any).inline = 1;\n return fn as T & { inline: 1 };\n};\n\n/**\n * Creates a match function for an inline scoped element from a regex.\n */\nexport const inlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for inline elements except links.\n */\nexport const simpleInlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for a block scoped element from a regex.\n */\nexport const blockRegex =\n (regex: RegExp) =>\n (source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return null;\n } else {\n return regex.exec(source);\n }\n };\n\n/**\n * Creates a match function from a regex, ignoring block/inline scope.\n */\nexport const anyScopeRegex = (\n fn: RegExp | ((source: string, state: ParseState) => RegExpMatchArray | null)\n) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (typeof fn === 'function') {\n return fn(source, state);\n }\n return fn.exec(source);\n });\n\n/**\n * Parse inline content (including links).\n */\nexport const parseInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n state.inline = true;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse simple inline content (no links).\n */\nexport const parseSimpleInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n\n state.inline = false;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseSimpleInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse block content.\n */\nexport const parseBlock = (\n parse: NestedParser,\n children: string,\n state: ParseState = {}\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline || false;\n state.inline = false;\n const normalizedChildren = trimEnd(children);\n const needsTerminator = /\\n\\n$/.test(normalizedChildren) === false;\n const blockInput = needsTerminator\n ? normalizedChildren.endsWith('\\n')\n ? `${normalizedChildren}\\n`\n : `${normalizedChildren}\\n\\n`\n : normalizedChildren;\n\n const result = parse(blockInput, state);\n state.inline = isCurrentlyInline;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseBlock: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Helper to parse capture group 2 as inline content.\n */\nexport const parseCaptureInline = (\n capture: RegExpMatchArray,\n parse: NestedParser,\n state: ParseState\n): { children: ParserResult[] } => {\n return {\n children: parseInline(parse, capture[2], state),\n };\n};\n\n/**\n * Helper that captures nothing (empty object).\n */\nexport const captureNothing = (): Record<string, never> => ({});\n\n/**\n * Helper that renders nothing (null).\n */\nexport const renderNothing = (): null => null;\n\n/**\n * Check if any regex in a list matches the input.\n */\nexport const some = (regexes: RegExp[], input: string): boolean => {\n for (let i = 0; i < regexes.length; i++) {\n if (regexes[i].test(input)) {\n return true;\n }\n }\n return false;\n};\n"],"mappings":"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"}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import{ATTRIBUTES_TO_SANITIZE as e,CAPTURE_LETTER_AFTER_HYPHEN as t,CR_NEWLINE_R as n,FORMFEED_R as r,HTML_CUSTOM_ATTR_R as i,INTERPOLATION_R as a,TABLE_CENTER_ALIGN as o,TABLE_LEFT_ALIGN as s,TABLE_RIGHT_ALIGN as c,TABLE_TRIM_PIPES as l,TAB_R as u,UNESCAPE_R as d}from"./constants.mjs";const f=e=>{let t=e.length;for(;t>0&&e[t-1]<=` `;)t--;return e.slice(0,t)},p=(e,t)=>e.startsWith(t),m=e=>{let t=e[0];return(t===`"`||t===`'`)&&e.length>=2&&e[e.length-1]===t?e.slice(1,-1):e},h=e=>e&&e.replace(d,`$1`),g=(...e)=>e.filter(Boolean).join(` `),_=(e,t,n)=>{let r=e,i=t.split(`.`);for(;i.length&&(r=r[i[0]],r!==void 0);)i.shift();return r??n},v=e=>e.replace(/[ÀÁÂÃÄÅàáâãä忯]/g,`a`).replace(/[çÇ]/g,`c`).replace(/[ðÐ]/g,`d`).replace(/[ÈÉÊËéèêë]/g,`e`).replace(/[ÏïÎîÍíÌì]/g,`i`).replace(/[Ññ]/g,`n`).replace(/[øØœŒÕõÔôÓóÒò]/g,`o`).replace(/[ÜüÛûÚúÙù]/g,`u`).replace(/[ŸÿÝý]/g,`y`).replace(/[^a-z0-9- ]/gi,``).replace(/ /gi,`-`).toLowerCase(),y=/(javascript|vbscript|data(?!:image)):/i,b=e=>{try{let t=decodeURIComponent(e).replace(/[^A-Za-z0-9/:]/g,``);if(y.test(t))return null}catch{return null}return e},x=e=>{let t=performance.now(),i=e.replace(n,`
|
|
1
|
+
import{ATTRIBUTES_TO_SANITIZE as e,CAPTURE_LETTER_AFTER_HYPHEN as t,CR_NEWLINE_R as n,FORMFEED_R as r,HTML_CUSTOM_ATTR_R as i,INTERPOLATION_R as a,TABLE_CENTER_ALIGN as o,TABLE_LEFT_ALIGN as s,TABLE_RIGHT_ALIGN as c,TABLE_TRIM_PIPES as l,TAB_R as u,UNESCAPE_R as d}from"./constants.mjs";const f=e=>{let t=e.length;for(;t>0&&e[t-1]<=` `;)t--;return e.slice(0,t)},p=(e,t)=>e.startsWith(t),m=e=>{let t=e[0];return(t===`"`||t===`'`)&&e.length>=2&&e[e.length-1]===t?e.slice(1,-1):e},h=e=>e&&e.replace(d,`$1`),g=(...e)=>e.filter(Boolean).join(` `),_=(e,t,n)=>{let r=e,i=t.split(`.`);for(;i.length&&(r=r[i[0]],r!==void 0);)i.shift();return r??n},v=e=>e.replace(/[ÀÁÂÃÄÅàáâãä忯]/g,`a`).replace(/[çÇ]/g,`c`).replace(/[ðÐ]/g,`d`).replace(/[ÈÉÊËéèêë]/g,`e`).replace(/[ÏïÎîÍíÌì]/g,`i`).replace(/[Ññ]/g,`n`).replace(/[øØœŒÕõÔôÓóÒò]/g,`o`).replace(/[ÜüÛûÚúÙù]/g,`u`).replace(/[ŸÿÝý]/g,`y`).replace(/[^a-z0-9- ]/gi,``).replace(/ /gi,`-`).toLowerCase(),y=/(javascript|vbscript|data(?!:image)):/i,b=e=>{try{let t=decodeURIComponent(e).replace(/[^A-Za-z0-9/:]/g,``);if(y.test(t))return console.warn(`Input contains an unsafe JavaScript/VBScript/data expression, it will not be rendered.`,t),null}catch{return console.warn(`Input could not be decoded due to malformed syntax or characters, it will not be rendered.`,e),null}return e},x=e=>{let t=performance.now(),i=e.replace(n,`
|
|
2
2
|
`).replace(r,``).replace(u,` `),a=performance.now()-t;return a>20&&console.log(`normalizeWhitespace: ${a.toFixed(3)}ms, source length: ${e.length}`),i},S=(e,t)=>{let n=performance.now();if(!t)return e;let r=e.split(`
|
|
3
3
|
`),i=!1,a=null,o=e=>e.match(/^\s*(`{3,}|~{3,})/),s=e=>{let t=o(e);if(!t)return;let n=t[1];i?a&&e.includes(a)&&(i=!1,a=null):(i=!0,a=n)},c=r.map(e=>{if(o(e)){let n=e.startsWith(t)?e.slice(t.length):e;return s(e),n}return i?e:e.startsWith(t)?e.slice(t.length):e}).join(`
|
|
4
4
|
`),l=performance.now()-n;return l>20&&console.log(`trimLeadingWhitespaceOutsideFences: ${l.toFixed(3)}ms, text length: ${e.length}, lines count: ${r.length}`),c},C=e=>(e.indexOf(`-`)!==-1&&e.match(i)===null&&(e=e.replace(t,(e,t)=>t.toUpperCase())),e),w=e=>{let t=performance.now(),n=[],r=``,i=!1,a=!1,o=``;if(!e)return n;for(let t=0;t<e.length;t++){let s=e[t];if((s===`"`||s===`'`)&&!i&&(a?s===o&&(a=!1,o=``):(a=!0,o=s)),s===`(`&&r.endsWith(`url`)?i=!0:s===`)`&&i&&(i=!1),s===`;`&&!a&&!i){let e=r.trim();if(e){let t=e.indexOf(`:`);if(t>0){let r=e.slice(0,t).trim(),i=e.slice(t+1).trim();n.push([r,i])}}r=``}else r+=s}let s=r.trim();if(s){let e=s.indexOf(`:`);if(e>0){let t=s.slice(0,e).trim(),r=s.slice(e+1).trim();n.push([t,r])}}let c=performance.now()-t;return c>20&&console.log(`parseStyleAttribute: ${c.toFixed(3)}ms, styleString length: ${e.length}, styles count: ${n.length}`),n},T=(t,n,r,i)=>n===`style`?w(r).reduce((e,[n,r])=>{let a=n.replace(/(-[a-z])/g,e=>e[1].toUpperCase());return e[a]=i(r,t,n),e},{}):e.indexOf(n)===-1?(r.match(a)&&(r=h(r.slice(1,r.length-1))),r===`true`?!0:r===`false`?!1:r):i(h(r),t,n),E=e=>c.test(e)?`right`:o.test(e)?`center`:(s.test(e),`left`),D=e=>e.replace(l,``).split(`|`).map(E),O=(e,t,n,r)=>{let i=performance.now(),a=n.inTable;n.inTable=!0;let o=[[]],s=``,c=()=>{if(!s)return;let e=o[o.length-1];e.push.apply(e,t(s,n)),s=``};e.trim().split(/(`[^`]*`|\\\||\|)/).filter(Boolean).forEach((e,t,n)=>{if(e.trim()===`|`&&(c(),r)){t!==0&&t!==n.length-1&&o.push([]);return}s+=e}),c(),n.inTable=a;let l=performance.now()-i;return l>20&&console.log(`parseTableRow: ${l.toFixed(3)}ms, source length: ${e.length}, cells count: ${o.length}`),o},k=(e,t,n)=>{let r=performance.now(),i=e.trim().split(`
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.mjs","names":[],"sources":["../../../src/markdown/utils.ts"],"sourcesContent":["import {\n ATTRIBUTES_TO_SANITIZE,\n CAPTURE_LETTER_AFTER_HYPHEN,\n CR_NEWLINE_R,\n DURATION_DELAY_TRIGGER,\n FORMFEED_R,\n HTML_CUSTOM_ATTR_R,\n INTERPOLATION_R,\n TAB_R,\n TABLE_CENTER_ALIGN,\n TABLE_LEFT_ALIGN,\n TABLE_RIGHT_ALIGN,\n TABLE_TRIM_PIPES,\n UNESCAPE_R,\n} from './constants';\nimport type { NestedParser, ParserResult, ParseState, Rule } from './types';\n\n// ============================================================================\n// STRING UTILITIES\n// ============================================================================\n\n/**\n * Trim trailing whitespace from a string.\n */\nexport const trimEnd = (str: string): string => {\n let end = str.length;\n\n while (end > 0 && str[end - 1] <= ' ') end--;\n\n return str.slice(0, end);\n};\n\n/**\n * Check if string starts with prefix.\n */\nexport const startsWith = (str: string, prefix: string): boolean => {\n return str.startsWith(prefix);\n};\n\n/**\n * Remove symmetrical leading and trailing quotes.\n */\nexport const unquote = (str: string): string => {\n const first = str[0];\n\n if (\n (first === '\"' || first === \"'\") &&\n str.length >= 2 &&\n str[str.length - 1] === first\n ) {\n return str.slice(1, -1);\n }\n\n return str;\n};\n\n/**\n * Unescape backslash-escaped characters.\n */\nexport const unescapeString = (rawString: string): string =>\n rawString ? rawString.replace(UNESCAPE_R, '$1') : rawString;\n\n/**\n * Join class names, filtering out falsy values.\n */\nexport const cx = (...args: any[]): string => args.filter(Boolean).join(' ');\n\n/**\n * Get a nested property from an object using dot notation.\n */\nexport const get = (src: any, path: string, fb?: any): any => {\n let ptr = src;\n const frags = path.split('.');\n\n while (frags.length) {\n ptr = ptr[frags[0]];\n\n if (ptr === undefined) break;\n else frags.shift();\n }\n\n return ptr ?? fb;\n};\n\n// ============================================================================\n// SLUGIFY\n// ============================================================================\n\n/**\n * Convert a string to a URL-safe slug.\n * Based on https://stackoverflow.com/a/18123682/1141611\n */\nexport const slugify = (str: string): string =>\n str\n .replace(/[ÀÁÂÃÄÅàáâãä忯]/g, 'a')\n .replace(/[çÇ]/g, 'c')\n .replace(/[ðÐ]/g, 'd')\n .replace(/[ÈÉÊËéèêë]/g, 'e')\n .replace(/[ÏïÎîÍíÌì]/g, 'i')\n .replace(/[Ññ]/g, 'n')\n .replace(/[øØœŒÕõÔôÓóÒò]/g, 'o')\n .replace(/[ÜüÛûÚúÙù]/g, 'u')\n .replace(/[ŸÿÝý]/g, 'y')\n .replace(/[^a-z0-9- ]/gi, '')\n .replace(/ /gi, '-')\n .toLowerCase();\n\n// ============================================================================\n// SANITIZER\n// ============================================================================\n\nconst SANITIZE_R = /(javascript|vbscript|data(?!:image)):/i;\n\n/**\n * Sanitize URLs to prevent XSS attacks.\n * Returns null if the URL is unsafe.\n */\nexport const sanitizer = (input: string): string | null => {\n try {\n const decoded = decodeURIComponent(input).replace(/[^A-Za-z0-9/:]/g, '');\n\n if (SANITIZE_R.test(decoded)) {\n if (process.env.NODE_ENV !== 'production') {\n console.warn(\n 'Input contains an unsafe JavaScript/VBScript/data expression, it will not be rendered.',\n decoded\n );\n }\n\n return null;\n }\n } catch (_e) {\n if (process.env.NODE_ENV !== 'production') {\n console.warn(\n 'Input could not be decoded due to malformed syntax or characters, it will not be rendered.',\n input\n );\n }\n\n // decodeURIComponent sometimes throws a URIError\n return null;\n }\n\n return input;\n};\n\n// ============================================================================\n// WHITESPACE NORMALIZATION\n// ============================================================================\n\n/**\n * Normalize whitespace in source string.\n */\nexport const normalizeWhitespace = (source: string): string => {\n const start = performance.now();\n const result = source\n .replace(CR_NEWLINE_R, '\\n')\n .replace(FORMFEED_R, '')\n .replace(TAB_R, ' ');\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `normalizeWhitespace: ${duration.toFixed(3)}ms, source length: ${source.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Safely remove a uniform leading indentation from lines, but do NOT touch\n * the content inside fenced code blocks (``` or ~~~).\n */\nexport const trimLeadingWhitespaceOutsideFences = (\n text: string,\n whitespace: string\n): string => {\n const start = performance.now();\n if (!whitespace) return text;\n\n const lines = text.split('\\n');\n let inFence = false;\n let fenceToken: string | null = null;\n\n const isFenceLine = (line: string): RegExpMatchArray | null =>\n line.match(/^\\s*(`{3,}|~{3,})/);\n\n const maybeToggleFence = (line: string): void => {\n const m = isFenceLine(line);\n\n if (!m) return;\n\n const token = m[1];\n\n if (!inFence) {\n inFence = true;\n fenceToken = token;\n } else if (fenceToken && line.includes(fenceToken)) {\n inFence = false;\n fenceToken = null;\n }\n };\n\n const out = lines.map((line) => {\n const fenceMatch = isFenceLine(line);\n if (fenceMatch) {\n const trimmedFenceLine = line.startsWith(whitespace)\n ? line.slice(whitespace.length)\n : line;\n maybeToggleFence(line);\n return trimmedFenceLine;\n }\n\n if (inFence) {\n return line;\n }\n\n return line.startsWith(whitespace) ? line.slice(whitespace.length) : line;\n });\n\n const result = out.join('\\n');\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `trimLeadingWhitespaceOutsideFences: ${duration.toFixed(3)}ms, text length: ${text.length}, lines count: ${lines.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Normalize HTML attribute key to JSX prop name.\n */\nexport const normalizeAttributeKey = (key: string): string => {\n const hyphenIndex = key.indexOf('-');\n\n if (hyphenIndex !== -1 && key.match(HTML_CUSTOM_ATTR_R) === null) {\n key = key.replace(CAPTURE_LETTER_AFTER_HYPHEN, (_, letter) => {\n return letter.toUpperCase();\n });\n }\n\n return key;\n};\n\ntype StyleTuple = [key: string, value: string];\n\n/**\n * Parse a CSS style string into an array of [key, value] tuples.\n */\nexport const parseStyleAttribute = (styleString: string): StyleTuple[] => {\n const start = performance.now();\n const styles: StyleTuple[] = [];\n let buffer = '';\n let inUrl = false;\n let inQuotes = false;\n let quoteChar: '\"' | \"'\" | '' = '';\n\n if (!styleString) return styles;\n\n for (let i = 0; i < styleString.length; i++) {\n const char = styleString[i];\n\n if ((char === '\"' || char === \"'\") && !inUrl) {\n if (!inQuotes) {\n inQuotes = true;\n quoteChar = char;\n } else if (char === quoteChar) {\n inQuotes = false;\n quoteChar = '';\n }\n }\n\n if (char === '(' && buffer.endsWith('url')) {\n inUrl = true;\n } else if (char === ')' && inUrl) {\n inUrl = false;\n }\n\n if (char === ';' && !inQuotes && !inUrl) {\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n buffer = '';\n } else {\n buffer += char;\n }\n }\n\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseStyleAttribute: ${duration.toFixed(3)}ms, styleString length: ${styleString.length}, styles count: ${styles.length}`\n );\n }\n\n return styles;\n};\n\n/**\n * Convert an attribute value to a Node prop value.\n */\nexport const attributeValueToNodePropValue = (\n tag: string,\n key: string,\n value: string,\n sanitizeUrlFn: (\n value: string,\n tag: string,\n attribute: string\n ) => string | null\n): any => {\n if (key === 'style') {\n return parseStyleAttribute(value).reduce(\n (styles, [styleKey, styleValue]) => {\n const camelCasedKey = styleKey.replace(/(-[a-z])/g, (substr) =>\n substr[1].toUpperCase()\n );\n\n (styles as Record<string, any>)[camelCasedKey] = sanitizeUrlFn(\n styleValue,\n tag,\n styleKey\n );\n\n return styles;\n },\n {} as Record<string, any>\n );\n } else if (ATTRIBUTES_TO_SANITIZE.indexOf(key) !== -1) {\n return sanitizeUrlFn(unescapeString(value), tag, key);\n } else if (value.match(INTERPOLATION_R)) {\n value = unescapeString(value.slice(1, value.length - 1));\n }\n\n if (value === 'true') {\n return true;\n } else if (value === 'false') {\n return false;\n }\n\n return value;\n};\n\n// ============================================================================\n// TABLE PARSING\n// ============================================================================\n\n/**\n * Parse table alignment from a separator row.\n */\nexport const parseTableAlignCapture = (\n alignCapture: string\n): 'left' | 'right' | 'center' => {\n if (TABLE_RIGHT_ALIGN.test(alignCapture)) {\n return 'right';\n } else if (TABLE_CENTER_ALIGN.test(alignCapture)) {\n return 'center';\n } else if (TABLE_LEFT_ALIGN.test(alignCapture)) {\n return 'left';\n }\n\n return 'left';\n};\n\n/**\n * Parse table alignment row.\n */\nexport const parseTableAlign = (\n source: string\n): ('left' | 'right' | 'center')[] => {\n const alignText = source.replace(TABLE_TRIM_PIPES, '').split('|');\n return alignText.map(parseTableAlignCapture);\n};\n\n/**\n * Parse a single table row.\n */\nexport const parseTableRow = (\n source: string,\n parse: NestedParser,\n state: ParseState,\n tableOutput: boolean\n): ParserResult[][] => {\n const start = performance.now();\n const prevInTable = state.inTable;\n\n state.inTable = true;\n\n const cells: ParserResult[][] = [[]];\n let acc = '';\n\n const flush = (): void => {\n if (!acc) return;\n\n const cell = cells[cells.length - 1];\n cell.push.apply(cell, parse(acc, state));\n acc = '';\n };\n\n source\n .trim()\n .split(/(`[^`]*`|\\\\\\||\\|)/)\n .filter(Boolean)\n .forEach((fragment, i, arr) => {\n if (fragment.trim() === '|') {\n flush();\n\n if (tableOutput) {\n if (i !== 0 && i !== arr.length - 1) {\n cells.push([]);\n }\n\n return;\n }\n }\n\n acc += fragment;\n });\n\n flush();\n\n state.inTable = prevInTable;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableRow: ${duration.toFixed(3)}ms, source length: ${source.length}, cells count: ${cells.length}`\n );\n }\n\n return cells;\n};\n\n/**\n * Parse table cells (multiple rows).\n */\nexport const parseTableCells = (\n source: string,\n parse: NestedParser,\n state: ParseState\n): ParserResult[][][] => {\n const start = performance.now();\n const rowsText = source.trim().split('\\n');\n\n const result = rowsText.map((rowText) =>\n parseTableRow(rowText, parse, state, true)\n );\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableCells: ${duration.toFixed(3)}ms, source length: ${source.length}, rows count: ${rowsText.length}`\n );\n }\n\n return result;\n};\n\n// ============================================================================\n// PARSING HELPERS\n// ============================================================================\n\n/**\n * Check if a rule qualifies for the current source and state.\n */\nexport const qualifies = (\n source: string,\n state: ParseState,\n qualify: NonNullable<Rule<any>['_qualify']>\n): boolean => {\n if (Array.isArray(qualify)) {\n for (let i = 0; i < qualify.length; i++) {\n if (startsWith(source, qualify[i])) return true;\n }\n\n return false;\n }\n\n return (qualify as (source: string, state: ParseState) => boolean)(\n source,\n state\n );\n};\n\n/**\n * Marks a matcher function as eligible for being run inside an inline context.\n */\nexport const allowInline = <T extends (...args: any[]) => any>(\n fn: T\n): T & { inline: 1 } => {\n (fn as any).inline = 1;\n return fn as T & { inline: 1 };\n};\n\n/**\n * Creates a match function for an inline scoped element from a regex.\n */\nexport const inlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for inline elements except links.\n */\nexport const simpleInlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for a block scoped element from a regex.\n */\nexport const blockRegex =\n (regex: RegExp) =>\n (source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return null;\n } else {\n return regex.exec(source);\n }\n };\n\n/**\n * Creates a match function from a regex, ignoring block/inline scope.\n */\nexport const anyScopeRegex = (\n fn: RegExp | ((source: string, state: ParseState) => RegExpMatchArray | null)\n) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (typeof fn === 'function') {\n return fn(source, state);\n }\n return fn.exec(source);\n });\n\n/**\n * Parse inline content (including links).\n */\nexport const parseInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n state.inline = true;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse simple inline content (no links).\n */\nexport const parseSimpleInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n\n state.inline = false;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseSimpleInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse block content.\n */\nexport const parseBlock = (\n parse: NestedParser,\n children: string,\n state: ParseState = {}\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline || false;\n state.inline = false;\n const normalizedChildren = trimEnd(children);\n const needsTerminator = /\\n\\n$/.test(normalizedChildren) === false;\n const blockInput = needsTerminator\n ? normalizedChildren.endsWith('\\n')\n ? `${normalizedChildren}\\n`\n : `${normalizedChildren}\\n\\n`\n : normalizedChildren;\n\n const result = parse(blockInput, state);\n state.inline = isCurrentlyInline;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseBlock: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Helper to parse capture group 2 as inline content.\n */\nexport const parseCaptureInline = (\n capture: RegExpMatchArray,\n parse: NestedParser,\n state: ParseState\n): { children: ParserResult[] } => {\n return {\n children: parseInline(parse, capture[2], state),\n };\n};\n\n/**\n * Helper that captures nothing (empty object).\n */\nexport const captureNothing = (): Record<string, never> => ({});\n\n/**\n * Helper that renders nothing (null).\n */\nexport const renderNothing = (): null => null;\n\n/**\n * Check if any regex in a list matches the input.\n */\nexport const some = (regexes: RegExp[], input: string): boolean => {\n for (let i = 0; i < regexes.length; i++) {\n if (regexes[i].test(input)) {\n return true;\n }\n }\n return false;\n};\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"utils.mjs","names":[],"sources":["../../../src/markdown/utils.ts"],"sourcesContent":["import {\n ATTRIBUTES_TO_SANITIZE,\n CAPTURE_LETTER_AFTER_HYPHEN,\n CR_NEWLINE_R,\n DURATION_DELAY_TRIGGER,\n FORMFEED_R,\n HTML_CUSTOM_ATTR_R,\n INTERPOLATION_R,\n TAB_R,\n TABLE_CENTER_ALIGN,\n TABLE_LEFT_ALIGN,\n TABLE_RIGHT_ALIGN,\n TABLE_TRIM_PIPES,\n UNESCAPE_R,\n} from './constants';\nimport type { NestedParser, ParserResult, ParseState, Rule } from './types';\n\n// ============================================================================\n// STRING UTILITIES\n// ============================================================================\n\n/**\n * Trim trailing whitespace from a string.\n */\nexport const trimEnd = (str: string): string => {\n let end = str.length;\n\n while (end > 0 && str[end - 1] <= ' ') end--;\n\n return str.slice(0, end);\n};\n\n/**\n * Check if string starts with prefix.\n */\nexport const startsWith = (str: string, prefix: string): boolean => {\n return str.startsWith(prefix);\n};\n\n/**\n * Remove symmetrical leading and trailing quotes.\n */\nexport const unquote = (str: string): string => {\n const first = str[0];\n\n if (\n (first === '\"' || first === \"'\") &&\n str.length >= 2 &&\n str[str.length - 1] === first\n ) {\n return str.slice(1, -1);\n }\n\n return str;\n};\n\n/**\n * Unescape backslash-escaped characters.\n */\nexport const unescapeString = (rawString: string): string =>\n rawString ? rawString.replace(UNESCAPE_R, '$1') : rawString;\n\n/**\n * Join class names, filtering out falsy values.\n */\nexport const cx = (...args: any[]): string => args.filter(Boolean).join(' ');\n\n/**\n * Get a nested property from an object using dot notation.\n */\nexport const get = (src: any, path: string, fb?: any): any => {\n let ptr = src;\n const frags = path.split('.');\n\n while (frags.length) {\n ptr = ptr[frags[0]];\n\n if (ptr === undefined) break;\n else frags.shift();\n }\n\n return ptr ?? fb;\n};\n\n// ============================================================================\n// SLUGIFY\n// ============================================================================\n\n/**\n * Convert a string to a URL-safe slug.\n * Based on https://stackoverflow.com/a/18123682/1141611\n */\nexport const slugify = (str: string): string =>\n str\n .replace(/[ÀÁÂÃÄÅàáâãä忯]/g, 'a')\n .replace(/[çÇ]/g, 'c')\n .replace(/[ðÐ]/g, 'd')\n .replace(/[ÈÉÊËéèêë]/g, 'e')\n .replace(/[ÏïÎîÍíÌì]/g, 'i')\n .replace(/[Ññ]/g, 'n')\n .replace(/[øØœŒÕõÔôÓóÒò]/g, 'o')\n .replace(/[ÜüÛûÚúÙù]/g, 'u')\n .replace(/[ŸÿÝý]/g, 'y')\n .replace(/[^a-z0-9- ]/gi, '')\n .replace(/ /gi, '-')\n .toLowerCase();\n\n// ============================================================================\n// SANITIZER\n// ============================================================================\n\nconst SANITIZE_R = /(javascript|vbscript|data(?!:image)):/i;\n\n/**\n * Sanitize URLs to prevent XSS attacks.\n * Returns null if the URL is unsafe.\n */\nexport const sanitizer = (input: string): string | null => {\n try {\n const decoded = decodeURIComponent(input).replace(/[^A-Za-z0-9/:]/g, '');\n\n if (SANITIZE_R.test(decoded)) {\n console.warn(\n 'Input contains an unsafe JavaScript/VBScript/data expression, it will not be rendered.',\n decoded\n );\n\n return null;\n }\n } catch (_e) {\n console.warn(\n 'Input could not be decoded due to malformed syntax or characters, it will not be rendered.',\n input\n );\n\n // decodeURIComponent sometimes throws a URIError\n return null;\n }\n\n return input;\n};\n\n// ============================================================================\n// WHITESPACE NORMALIZATION\n// ============================================================================\n\n/**\n * Normalize whitespace in source string.\n */\nexport const normalizeWhitespace = (source: string): string => {\n const start = performance.now();\n const result = source\n .replace(CR_NEWLINE_R, '\\n')\n .replace(FORMFEED_R, '')\n .replace(TAB_R, ' ');\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `normalizeWhitespace: ${duration.toFixed(3)}ms, source length: ${source.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Safely remove a uniform leading indentation from lines, but do NOT touch\n * the content inside fenced code blocks (``` or ~~~).\n */\nexport const trimLeadingWhitespaceOutsideFences = (\n text: string,\n whitespace: string\n): string => {\n const start = performance.now();\n if (!whitespace) return text;\n\n const lines = text.split('\\n');\n let inFence = false;\n let fenceToken: string | null = null;\n\n const isFenceLine = (line: string): RegExpMatchArray | null =>\n line.match(/^\\s*(`{3,}|~{3,})/);\n\n const maybeToggleFence = (line: string): void => {\n const m = isFenceLine(line);\n\n if (!m) return;\n\n const token = m[1];\n\n if (!inFence) {\n inFence = true;\n fenceToken = token;\n } else if (fenceToken && line.includes(fenceToken)) {\n inFence = false;\n fenceToken = null;\n }\n };\n\n const out = lines.map((line) => {\n const fenceMatch = isFenceLine(line);\n if (fenceMatch) {\n const trimmedFenceLine = line.startsWith(whitespace)\n ? line.slice(whitespace.length)\n : line;\n maybeToggleFence(line);\n return trimmedFenceLine;\n }\n\n if (inFence) {\n return line;\n }\n\n return line.startsWith(whitespace) ? line.slice(whitespace.length) : line;\n });\n\n const result = out.join('\\n');\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `trimLeadingWhitespaceOutsideFences: ${duration.toFixed(3)}ms, text length: ${text.length}, lines count: ${lines.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Normalize HTML attribute key to JSX prop name.\n */\nexport const normalizeAttributeKey = (key: string): string => {\n const hyphenIndex = key.indexOf('-');\n\n if (hyphenIndex !== -1 && key.match(HTML_CUSTOM_ATTR_R) === null) {\n key = key.replace(CAPTURE_LETTER_AFTER_HYPHEN, (_, letter) => {\n return letter.toUpperCase();\n });\n }\n\n return key;\n};\n\ntype StyleTuple = [key: string, value: string];\n\n/**\n * Parse a CSS style string into an array of [key, value] tuples.\n */\nexport const parseStyleAttribute = (styleString: string): StyleTuple[] => {\n const start = performance.now();\n const styles: StyleTuple[] = [];\n let buffer = '';\n let inUrl = false;\n let inQuotes = false;\n let quoteChar: '\"' | \"'\" | '' = '';\n\n if (!styleString) return styles;\n\n for (let i = 0; i < styleString.length; i++) {\n const char = styleString[i];\n\n if ((char === '\"' || char === \"'\") && !inUrl) {\n if (!inQuotes) {\n inQuotes = true;\n quoteChar = char;\n } else if (char === quoteChar) {\n inQuotes = false;\n quoteChar = '';\n }\n }\n\n if (char === '(' && buffer.endsWith('url')) {\n inUrl = true;\n } else if (char === ')' && inUrl) {\n inUrl = false;\n }\n\n if (char === ';' && !inQuotes && !inUrl) {\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n buffer = '';\n } else {\n buffer += char;\n }\n }\n\n const declaration = buffer.trim();\n\n if (declaration) {\n const colonIndex = declaration.indexOf(':');\n if (colonIndex > 0) {\n const key = declaration.slice(0, colonIndex).trim();\n const value = declaration.slice(colonIndex + 1).trim();\n styles.push([key, value]);\n }\n }\n\n const duration = performance.now() - start;\n\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseStyleAttribute: ${duration.toFixed(3)}ms, styleString length: ${styleString.length}, styles count: ${styles.length}`\n );\n }\n\n return styles;\n};\n\n/**\n * Convert an attribute value to a Node prop value.\n */\nexport const attributeValueToNodePropValue = (\n tag: string,\n key: string,\n value: string,\n sanitizeUrlFn: (\n value: string,\n tag: string,\n attribute: string\n ) => string | null\n): any => {\n if (key === 'style') {\n return parseStyleAttribute(value).reduce(\n (styles, [styleKey, styleValue]) => {\n const camelCasedKey = styleKey.replace(/(-[a-z])/g, (substr) =>\n substr[1].toUpperCase()\n );\n\n (styles as Record<string, any>)[camelCasedKey] = sanitizeUrlFn(\n styleValue,\n tag,\n styleKey\n );\n\n return styles;\n },\n {} as Record<string, any>\n );\n } else if (ATTRIBUTES_TO_SANITIZE.indexOf(key) !== -1) {\n return sanitizeUrlFn(unescapeString(value), tag, key);\n } else if (value.match(INTERPOLATION_R)) {\n value = unescapeString(value.slice(1, value.length - 1));\n }\n\n if (value === 'true') {\n return true;\n } else if (value === 'false') {\n return false;\n }\n\n return value;\n};\n\n// ============================================================================\n// TABLE PARSING\n// ============================================================================\n\n/**\n * Parse table alignment from a separator row.\n */\nexport const parseTableAlignCapture = (\n alignCapture: string\n): 'left' | 'right' | 'center' => {\n if (TABLE_RIGHT_ALIGN.test(alignCapture)) {\n return 'right';\n } else if (TABLE_CENTER_ALIGN.test(alignCapture)) {\n return 'center';\n } else if (TABLE_LEFT_ALIGN.test(alignCapture)) {\n return 'left';\n }\n\n return 'left';\n};\n\n/**\n * Parse table alignment row.\n */\nexport const parseTableAlign = (\n source: string\n): ('left' | 'right' | 'center')[] => {\n const alignText = source.replace(TABLE_TRIM_PIPES, '').split('|');\n return alignText.map(parseTableAlignCapture);\n};\n\n/**\n * Parse a single table row.\n */\nexport const parseTableRow = (\n source: string,\n parse: NestedParser,\n state: ParseState,\n tableOutput: boolean\n): ParserResult[][] => {\n const start = performance.now();\n const prevInTable = state.inTable;\n\n state.inTable = true;\n\n const cells: ParserResult[][] = [[]];\n let acc = '';\n\n const flush = (): void => {\n if (!acc) return;\n\n const cell = cells[cells.length - 1];\n cell.push.apply(cell, parse(acc, state));\n acc = '';\n };\n\n source\n .trim()\n .split(/(`[^`]*`|\\\\\\||\\|)/)\n .filter(Boolean)\n .forEach((fragment, i, arr) => {\n if (fragment.trim() === '|') {\n flush();\n\n if (tableOutput) {\n if (i !== 0 && i !== arr.length - 1) {\n cells.push([]);\n }\n\n return;\n }\n }\n\n acc += fragment;\n });\n\n flush();\n\n state.inTable = prevInTable;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableRow: ${duration.toFixed(3)}ms, source length: ${source.length}, cells count: ${cells.length}`\n );\n }\n\n return cells;\n};\n\n/**\n * Parse table cells (multiple rows).\n */\nexport const parseTableCells = (\n source: string,\n parse: NestedParser,\n state: ParseState\n): ParserResult[][][] => {\n const start = performance.now();\n const rowsText = source.trim().split('\\n');\n\n const result = rowsText.map((rowText) =>\n parseTableRow(rowText, parse, state, true)\n );\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseTableCells: ${duration.toFixed(3)}ms, source length: ${source.length}, rows count: ${rowsText.length}`\n );\n }\n\n return result;\n};\n\n// ============================================================================\n// PARSING HELPERS\n// ============================================================================\n\n/**\n * Check if a rule qualifies for the current source and state.\n */\nexport const qualifies = (\n source: string,\n state: ParseState,\n qualify: NonNullable<Rule<any>['_qualify']>\n): boolean => {\n if (Array.isArray(qualify)) {\n for (let i = 0; i < qualify.length; i++) {\n if (startsWith(source, qualify[i])) return true;\n }\n\n return false;\n }\n\n return (qualify as (source: string, state: ParseState) => boolean)(\n source,\n state\n );\n};\n\n/**\n * Marks a matcher function as eligible for being run inside an inline context.\n */\nexport const allowInline = <T extends (...args: any[]) => any>(\n fn: T\n): T & { inline: 1 } => {\n (fn as any).inline = 1;\n return fn as T & { inline: 1 };\n};\n\n/**\n * Creates a match function for an inline scoped element from a regex.\n */\nexport const inlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for inline elements except links.\n */\nexport const simpleInlineRegex = (regex: RegExp) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return regex.exec(source);\n } else {\n return null;\n }\n });\n\n/**\n * Creates a match function for a block scoped element from a regex.\n */\nexport const blockRegex =\n (regex: RegExp) =>\n (source: string, state: ParseState): RegExpMatchArray | null => {\n if (state.inline || state.simple) {\n return null;\n } else {\n return regex.exec(source);\n }\n };\n\n/**\n * Creates a match function from a regex, ignoring block/inline scope.\n */\nexport const anyScopeRegex = (\n fn: RegExp | ((source: string, state: ParseState) => RegExpMatchArray | null)\n) =>\n allowInline((source: string, state: ParseState): RegExpMatchArray | null => {\n if (typeof fn === 'function') {\n return fn(source, state);\n }\n return fn.exec(source);\n });\n\n/**\n * Parse inline content (including links).\n */\nexport const parseInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n state.inline = true;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse simple inline content (no links).\n */\nexport const parseSimpleInline = (\n parse: NestedParser,\n children: string,\n state: ParseState\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline ?? false;\n const isCurrentlySimple = state.simple ?? false;\n\n state.inline = false;\n state.simple = true;\n const result = parse(children, state);\n state.inline = isCurrentlyInline;\n state.simple = isCurrentlySimple;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseSimpleInline: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Parse block content.\n */\nexport const parseBlock = (\n parse: NestedParser,\n children: string,\n state: ParseState = {}\n): ParserResult[] => {\n const start = performance.now();\n const isCurrentlyInline = state.inline || false;\n state.inline = false;\n const normalizedChildren = trimEnd(children);\n const needsTerminator = /\\n\\n$/.test(normalizedChildren) === false;\n const blockInput = needsTerminator\n ? normalizedChildren.endsWith('\\n')\n ? `${normalizedChildren}\\n`\n : `${normalizedChildren}\\n\\n`\n : normalizedChildren;\n\n const result = parse(blockInput, state);\n state.inline = isCurrentlyInline;\n\n const duration = performance.now() - start;\n if (duration > DURATION_DELAY_TRIGGER) {\n console.log(\n `parseBlock: ${duration.toFixed(3)}ms, children length: ${children.length}, result count: ${result.length}`\n );\n }\n\n return result;\n};\n\n/**\n * Helper to parse capture group 2 as inline content.\n */\nexport const parseCaptureInline = (\n capture: RegExpMatchArray,\n parse: NestedParser,\n state: ParseState\n): { children: ParserResult[] } => {\n return {\n children: parseInline(parse, capture[2], state),\n };\n};\n\n/**\n * Helper that captures nothing (empty object).\n */\nexport const captureNothing = (): Record<string, never> => ({});\n\n/**\n * Helper that renders nothing (null).\n */\nexport const renderNothing = (): null => null;\n\n/**\n * Check if any regex in a list matches the input.\n */\nexport const some = (regexes: RegExp[], input: string): boolean => {\n for (let i = 0; i < regexes.length; i++) {\n if (regexes[i].test(input)) {\n return true;\n }\n }\n return false;\n};\n"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.d.ts","names":[],"sources":["../../../src/markdown/utils.ts"],"mappings":";;;;;AAwBA;cAAa,OAAA,GAAW,GAAA;;;;cAWX,UAAA,GAAc,GAAA,UAAa,MAAA;;;;cAO3B,OAAA,GAAW,GAAA;AAAxB;;;AAAA,cAiBa,cAAA,GAAkB,SAAA;;AAA/B;;cAMa,EAAA,MAAS,IAAA;;;AAAtB;cAKa,GAAA,GAAO,GAAA,OAAU,IAAA,UAAc,EAAA;;;;AAA5C;cAsBa,OAAA,GAAW,GAAA;;;;;cAyBX,SAAA,GAAa,KAAA;;;AAzB1B;
|
|
1
|
+
{"version":3,"file":"utils.d.ts","names":[],"sources":["../../../src/markdown/utils.ts"],"mappings":";;;;;AAwBA;cAAa,OAAA,GAAW,GAAA;;;;cAWX,UAAA,GAAc,GAAA,UAAa,MAAA;;;;cAO3B,OAAA,GAAW,GAAA;AAAxB;;;AAAA,cAiBa,cAAA,GAAkB,SAAA;;AAA/B;;cAMa,EAAA,MAAS,IAAA;;;AAAtB;cAKa,GAAA,GAAO,GAAA,OAAU,IAAA,UAAc,EAAA;;;;AAA5C;cAsBa,OAAA,GAAW,GAAA;;;;;cAyBX,SAAA,GAAa,KAAA;;;AAzB1B;cAyDa,mBAAA,GAAuB,MAAA;;;;AAhCpC;cAsDa,kCAAA,GACX,IAAA,UACA,UAAA;;;;cA4DW,qBAAA,GAAyB,GAAA;AAAA,KAYjC,UAAA,IAAc,GAAA,UAAa,KAAA;;;;cAKnB,mBAAA,GAAuB,WAAA,aAAsB,UAAA;;;;cAwE7C,6BAAA,GACX,GAAA,UACA,GAAA,UACA,KAAA,UACA,aAAA,GACE,KAAA,UACA,GAAA,UACA,SAAA;AAhGJ;;;AAAA,cA0Ia,sBAAA,GACX,YAAA;;AAjIA;;cAiJW,eAAA,GACX,MAAA;;;AA3IF;cAoJa,aAAA,GACX,MAAA,UACA,KAAA,EAAO,YAAA,EACP,KAAA,EAAO,UAAA,EACP,WAAA,cACC,YAAA;;;;cAsDU,eAAA,GACX,MAAA,UACA,KAAA,EAAO,YAAA,EACP,KAAA,EAAO,UAAA,KACN,YAAA;;;;cAyBU,SAAA,GACX,MAAA,UACA,KAAA,EAAO,UAAA,EACP,OAAA,EAAS,WAAA,CAAY,IAAA;;;;cAmBV,WAAA,iBAA6B,IAAA,iBACxC,EAAA,EAAI,CAAA,KACH,CAAA;EAAM,MAAA;AAAA;;;AA3IT;cAmJa,WAAA,GAAe,KAAA,EAAO,MAAA,OAAM,MAAA,UACZ,KAAA,EAAS,UAAA,KAAa,gBAAA;;;;AAnInD;;cA8Ia,iBAAA,GAAqB,KAAA,EAAO,MAAA,OAAM,MAAA,UAClB,KAAA,EAAS,UAAA,KAAa,gBAAA;;;AArInD;;;AAAA,cAgJa,UAAA,GACV,KAAA,EAAO,MAAA,MACP,MAAA,UAAgB,KAAA,EAAO,UAAA,KAAa,gBAAA;;;;cAW1B,aAAA,GACX,EAAA,EAAI,MAAA,KAAW,MAAA,UAAgB,KAAA,EAAO,UAAA,KAAe,gBAAA,eAAwB,MAAA,UAElD,KAAA,EAAS,UAAA,KAAa,gBAAA;;;;;;cAUtC,WAAA,GACX,KAAA,EAAO,YAAA,EACP,QAAA,UACA,KAAA,EAAO,UAAA,KACN,YAAA;;;;cAuBU,iBAAA,GACX,KAAA,EAAO,YAAA,EACP,QAAA,UACA,KAAA,EAAO,UAAA,KACN,YAAA;;;;cAwBU,UAAA,GACX,KAAA,EAAO,YAAA,EACP,QAAA,UACA,KAAA,GAAO,UAAA,KACN,YAAA;;;;cA4BU,kBAAA,GACX,OAAA,EAAS,gBAAA,EACT,KAAA,EAAO,YAAA,EACP,KAAA,EAAO,UAAA;EACJ,QAAA,EAAU,YAAA;AAAA;;;;cASF,cAAA,QAAqB,MAAA;;AAtLlC;;cA2La,aAAA;;;;cAKA,IAAA,GAAQ,OAAA,EAAS,MAAA,IAAU,KAAA"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@intlayer/core",
|
|
3
|
-
"version": "8.3.0-canary.
|
|
3
|
+
"version": "8.3.0-canary.4",
|
|
4
4
|
"private": false,
|
|
5
5
|
"description": "Includes core Intlayer functions like translation, dictionary, and utility functions shared across multiple packages.",
|
|
6
6
|
"keywords": [
|
|
@@ -168,11 +168,11 @@
|
|
|
168
168
|
"typecheck": "tsc --noEmit --project tsconfig.types.json"
|
|
169
169
|
},
|
|
170
170
|
"dependencies": {
|
|
171
|
-
"@intlayer/api": "8.3.0-canary.
|
|
172
|
-
"@intlayer/config": "8.3.0-canary.
|
|
173
|
-
"@intlayer/dictionaries-entry": "8.3.0-canary.
|
|
174
|
-
"@intlayer/types": "8.3.0-canary.
|
|
175
|
-
"@intlayer/unmerged-dictionaries-entry": "8.3.0-canary.
|
|
171
|
+
"@intlayer/api": "8.3.0-canary.4",
|
|
172
|
+
"@intlayer/config": "8.3.0-canary.4",
|
|
173
|
+
"@intlayer/dictionaries-entry": "8.3.0-canary.4",
|
|
174
|
+
"@intlayer/types": "8.3.0-canary.4",
|
|
175
|
+
"@intlayer/unmerged-dictionaries-entry": "8.3.0-canary.4",
|
|
176
176
|
"defu": "6.1.4"
|
|
177
177
|
},
|
|
178
178
|
"devDependencies": {
|