@djangocfg/ui-tools 2.1.334 → 2.1.336

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.
Files changed (196) hide show
  1. package/README.md +68 -2
  2. package/dist/ChatRoot-IIYQEWUU.mjs +5 -0
  3. package/dist/ChatRoot-IIYQEWUU.mjs.map +1 -0
  4. package/dist/ChatRoot-PNNGQCYF.css +7 -0
  5. package/dist/ChatRoot-PNNGQCYF.css.map +1 -0
  6. package/dist/ChatRoot-UUKTYM4N.cjs +14 -0
  7. package/dist/ChatRoot-UUKTYM4N.cjs.map +1 -0
  8. package/dist/{CronScheduler.client-3O3VU4CI.mjs → CronScheduler.client-DLMXCPAJ.mjs} +4 -4
  9. package/dist/{CronScheduler.client-3O3VU4CI.mjs.map → CronScheduler.client-DLMXCPAJ.mjs.map} +1 -1
  10. package/dist/{CronScheduler.client-A4GO6YBY.cjs → CronScheduler.client-WEJF4PWQ.cjs} +14 -14
  11. package/dist/{CronScheduler.client-A4GO6YBY.cjs.map → CronScheduler.client-WEJF4PWQ.cjs.map} +1 -1
  12. package/dist/{DocsLayout-XLDB6CJ2.cjs → DocsLayout-N5ZJZPBY.cjs} +200 -199
  13. package/dist/DocsLayout-N5ZJZPBY.cjs.map +1 -0
  14. package/dist/{DocsLayout-CTJINVBM.mjs → DocsLayout-VFPPNKSQ.mjs} +7 -6
  15. package/dist/DocsLayout-VFPPNKSQ.mjs.map +1 -0
  16. package/dist/JsonSchemaForm-DD7CLRIG.cjs +13 -0
  17. package/dist/{JsonSchemaForm-OSPUUUHM.cjs.map → JsonSchemaForm-DD7CLRIG.cjs.map} +1 -1
  18. package/dist/JsonSchemaForm-XKUIVELK.mjs +4 -0
  19. package/dist/{JsonSchemaForm-TSLX2GRO.mjs.map → JsonSchemaForm-XKUIVELK.mjs.map} +1 -1
  20. package/dist/JsonTree-55625VVH.mjs +5 -0
  21. package/dist/{JsonTree-F27RMYSI.cjs.map → JsonTree-55625VVH.mjs.map} +1 -1
  22. package/dist/JsonTree-DCM5QGWF.cjs +11 -0
  23. package/dist/{JsonTree-QTJYSHCV.mjs.map → JsonTree-DCM5QGWF.cjs.map} +1 -1
  24. package/dist/{LottiePlayer.client-6WVWDO75.cjs → LottiePlayer.client-2S7ISJ2S.cjs} +6 -6
  25. package/dist/{LottiePlayer.client-6WVWDO75.cjs.map → LottiePlayer.client-2S7ISJ2S.cjs.map} +1 -1
  26. package/dist/{LottiePlayer.client-B4I6WNZM.mjs → LottiePlayer.client-5LDSSJWS.mjs} +4 -4
  27. package/dist/{LottiePlayer.client-B4I6WNZM.mjs.map → LottiePlayer.client-5LDSSJWS.mjs.map} +1 -1
  28. package/dist/{MapContainer-RYG4HPH4.cjs → MapContainer-76YL2JXL.cjs} +8 -8
  29. package/dist/{MapContainer-RYG4HPH4.cjs.map → MapContainer-76YL2JXL.cjs.map} +1 -1
  30. package/dist/{MapContainer-GXQLP5WY.mjs → MapContainer-7HXBI3OH.mjs} +3 -3
  31. package/dist/{MapContainer-GXQLP5WY.mjs.map → MapContainer-7HXBI3OH.mjs.map} +1 -1
  32. package/dist/{Mermaid.client-SXRRI2YW.mjs → Mermaid.client-NL4SVR7F.mjs} +4 -4
  33. package/dist/{Mermaid.client-SXRRI2YW.mjs.map → Mermaid.client-NL4SVR7F.mjs.map} +1 -1
  34. package/dist/{Mermaid.client-W76R5AKJ.cjs → Mermaid.client-NNTI6DFX.cjs} +26 -26
  35. package/dist/{Mermaid.client-W76R5AKJ.cjs.map → Mermaid.client-NNTI6DFX.cjs.map} +1 -1
  36. package/dist/Player-BRV7XTWR.mjs +4 -0
  37. package/dist/{Player-M3GC3VPE.mjs.map → Player-BRV7XTWR.mjs.map} +1 -1
  38. package/dist/Player-PM7F7DD7.cjs +13 -0
  39. package/dist/{Player-ZL2X5LGG.cjs.map → Player-PM7F7DD7.cjs.map} +1 -1
  40. package/dist/{PrettyCode.client-RPDIE5CH.cjs → PrettyCode.client-KOHDVPPN.cjs} +13 -13
  41. package/dist/{PrettyCode.client-RPDIE5CH.cjs.map → PrettyCode.client-KOHDVPPN.cjs.map} +1 -1
  42. package/dist/{PrettyCode.client-SPMTQEG4.mjs → PrettyCode.client-ZGYGKE7G.mjs} +4 -4
  43. package/dist/{PrettyCode.client-SPMTQEG4.mjs.map → PrettyCode.client-ZGYGKE7G.mjs.map} +1 -1
  44. package/dist/TreeRoot-N72OYKXU.cjs +19 -0
  45. package/dist/{TreeRoot-A3J65L6F.mjs.map → TreeRoot-N72OYKXU.cjs.map} +1 -1
  46. package/dist/TreeRoot-VGAIXCUA.mjs +4 -0
  47. package/dist/{TreeRoot-DSK5JILT.cjs.map → TreeRoot-VGAIXCUA.mjs.map} +1 -1
  48. package/dist/chunk-2ZLKZ5VR.mjs +631 -0
  49. package/dist/chunk-2ZLKZ5VR.mjs.map +1 -0
  50. package/dist/{chunk-LFWQ36LJ.mjs → chunk-5G5YBFS6.mjs} +4 -4
  51. package/dist/{chunk-LFWQ36LJ.mjs.map → chunk-5G5YBFS6.mjs.map} +1 -1
  52. package/dist/{chunk-IHAY6FO6.cjs → chunk-5I5QNGUG.cjs} +17 -17
  53. package/dist/{chunk-IHAY6FO6.cjs.map → chunk-5I5QNGUG.cjs.map} +1 -1
  54. package/dist/{chunk-F2CMIIOH.cjs → chunk-76NNDZH6.cjs} +42 -42
  55. package/dist/{chunk-F2CMIIOH.cjs.map → chunk-76NNDZH6.cjs.map} +1 -1
  56. package/dist/chunk-B5AWZOHJ.cjs +649 -0
  57. package/dist/chunk-B5AWZOHJ.cjs.map +1 -0
  58. package/dist/{chunk-KR6B3LVY.mjs → chunk-B6IR5KSC.mjs} +3 -3
  59. package/dist/{chunk-KR6B3LVY.mjs.map → chunk-B6IR5KSC.mjs.map} +1 -1
  60. package/dist/{chunk-5LBDYFWH.mjs → chunk-C6GXVH5J.mjs} +3 -3
  61. package/dist/{chunk-5LBDYFWH.mjs.map → chunk-C6GXVH5J.mjs.map} +1 -1
  62. package/dist/{chunk-4IW7GZFQ.cjs → chunk-FEN5S772.cjs} +74 -48
  63. package/dist/chunk-FEN5S772.cjs.map +1 -0
  64. package/dist/{chunk-2SMCH62O.cjs → chunk-FP2RLYQZ.cjs} +11 -11
  65. package/dist/{chunk-2SMCH62O.cjs.map → chunk-FP2RLYQZ.cjs.map} +1 -1
  66. package/dist/{chunk-MOME6KYD.mjs → chunk-G5IEC7SR.mjs} +3 -3
  67. package/dist/{chunk-MOME6KYD.mjs.map → chunk-G5IEC7SR.mjs.map} +1 -1
  68. package/dist/{chunk-EXGXUK2N.mjs → chunk-GYIO7W7M.mjs} +41 -15
  69. package/dist/chunk-GYIO7W7M.mjs.map +1 -0
  70. package/dist/{chunk-3Z3A7FHA.cjs → chunk-IEEAENLX.cjs} +48 -48
  71. package/dist/{chunk-3Z3A7FHA.cjs.map → chunk-IEEAENLX.cjs.map} +1 -1
  72. package/dist/{chunk-DFTVB66S.cjs → chunk-KNDLV4PI.cjs} +85 -85
  73. package/dist/{chunk-DFTVB66S.cjs.map → chunk-KNDLV4PI.cjs.map} +1 -1
  74. package/dist/{chunk-SSUOENAZ.mjs → chunk-KNEQRUBA.mjs} +3 -3
  75. package/dist/{chunk-SSUOENAZ.mjs.map → chunk-KNEQRUBA.mjs.map} +1 -1
  76. package/dist/chunk-KRETIZU6.mjs +2218 -0
  77. package/dist/chunk-KRETIZU6.mjs.map +1 -0
  78. package/dist/{chunk-CGILA3WO.mjs → chunk-N2XQF2OL.mjs} +5 -3
  79. package/dist/{chunk-CGILA3WO.mjs.map → chunk-N2XQF2OL.mjs.map} +1 -1
  80. package/dist/{chunk-EUADAUBQ.mjs → chunk-N4MZYNR4.mjs} +4 -4
  81. package/dist/{chunk-EUADAUBQ.mjs.map → chunk-N4MZYNR4.mjs.map} +1 -1
  82. package/dist/chunk-NRXYYO5V.cjs +2257 -0
  83. package/dist/chunk-NRXYYO5V.cjs.map +1 -0
  84. package/dist/{chunk-GGKGH5PM.mjs → chunk-OBRSGM64.mjs} +4 -4
  85. package/dist/{chunk-GGKGH5PM.mjs.map → chunk-OBRSGM64.mjs.map} +1 -1
  86. package/dist/{chunk-6JTB2X72.mjs → chunk-ODO4GMW7.mjs} +3 -3
  87. package/dist/{chunk-6JTB2X72.mjs.map → chunk-ODO4GMW7.mjs.map} +1 -1
  88. package/dist/{chunk-WGEGR3DF.cjs → chunk-OLISEQHS.cjs} +5 -2
  89. package/dist/{chunk-WGEGR3DF.cjs.map → chunk-OLISEQHS.cjs.map} +1 -1
  90. package/dist/{chunk-PZKAH7WQ.mjs → chunk-PVAX67JG.mjs} +3 -3
  91. package/dist/{chunk-PZKAH7WQ.mjs.map → chunk-PVAX67JG.mjs.map} +1 -1
  92. package/dist/{chunk-PRPG2T2E.cjs → chunk-QJ6GTUCO.cjs} +6 -6
  93. package/dist/{chunk-PRPG2T2E.cjs.map → chunk-QJ6GTUCO.cjs.map} +1 -1
  94. package/dist/chunk-QW4RBGHN.cjs +961 -0
  95. package/dist/chunk-QW4RBGHN.cjs.map +1 -0
  96. package/dist/{chunk-33AMWFBZ.cjs → chunk-SGP7V2UW.cjs} +15 -15
  97. package/dist/{chunk-33AMWFBZ.cjs.map → chunk-SGP7V2UW.cjs.map} +1 -1
  98. package/dist/{chunk-FX2QFYWF.mjs → chunk-VWQ5WOIL.mjs} +3 -3
  99. package/dist/{chunk-FX2QFYWF.mjs.map → chunk-VWQ5WOIL.mjs.map} +1 -1
  100. package/dist/{chunk-ZLQHUZDU.cjs → chunk-YDPDTOSP.cjs} +139 -139
  101. package/dist/{chunk-ZLQHUZDU.cjs.map → chunk-YDPDTOSP.cjs.map} +1 -1
  102. package/dist/{chunk-77HQWEQ6.cjs → chunk-YW5IVWHQ.cjs} +33 -33
  103. package/dist/{chunk-77HQWEQ6.cjs.map → chunk-YW5IVWHQ.cjs.map} +1 -1
  104. package/dist/{chunk-YXBOAGIM.cjs → chunk-YXZ6GU7H.cjs} +7 -7
  105. package/dist/{chunk-YXBOAGIM.cjs.map → chunk-YXZ6GU7H.cjs.map} +1 -1
  106. package/dist/{chunk-62Y65TGK.mjs → chunk-ZUFTH5IR.mjs} +8 -631
  107. package/dist/chunk-ZUFTH5IR.mjs.map +1 -0
  108. package/dist/components-EHOGXATG.cjs +22 -0
  109. package/dist/{components-5UXYNAKR.cjs.map → components-EHOGXATG.cjs.map} +1 -1
  110. package/dist/components-MQ6DR7TX.cjs +26 -0
  111. package/dist/{components-CFXOEVPN.mjs.map → components-MQ6DR7TX.cjs.map} +1 -1
  112. package/dist/components-XRX7QGLB.mjs +5 -0
  113. package/dist/{components-WYEZL5TE.cjs.map → components-XRX7QGLB.mjs.map} +1 -1
  114. package/dist/components-YATKRWLH.mjs +5 -0
  115. package/dist/{components-ZAGG2PBO.mjs.map → components-YATKRWLH.mjs.map} +1 -1
  116. package/dist/file-icon/index.cjs +6 -6
  117. package/dist/file-icon/index.mjs +1 -1
  118. package/dist/index.cjs +735 -215
  119. package/dist/index.cjs.map +1 -1
  120. package/dist/index.d.cts +972 -39
  121. package/dist/index.d.ts +972 -39
  122. package/dist/index.mjs +387 -31
  123. package/dist/index.mjs.map +1 -1
  124. package/dist/tree/index.cjs +38 -38
  125. package/dist/tree/index.d.cts +2 -2
  126. package/dist/tree/index.d.ts +2 -2
  127. package/dist/tree/index.mjs +3 -3
  128. package/package.json +6 -6
  129. package/src/index.ts +5 -0
  130. package/src/stories/index.ts +3 -1
  131. package/src/tools/Chat/Chat.story.tsx +1006 -0
  132. package/src/tools/Chat/README.md +528 -0
  133. package/src/tools/Chat/components/Attachments.tsx +192 -0
  134. package/src/tools/Chat/components/ChatRoot.tsx +201 -0
  135. package/src/tools/Chat/components/Composer.tsx +134 -0
  136. package/src/tools/Chat/components/EmptyState.tsx +47 -0
  137. package/src/tools/Chat/components/ErrorBanner.tsx +47 -0
  138. package/src/tools/Chat/components/JumpToLatest.tsx +30 -0
  139. package/src/tools/Chat/components/MessageActions.tsx +72 -0
  140. package/src/tools/Chat/components/MessageBubble.tsx +228 -0
  141. package/src/tools/Chat/components/MessageList.tsx +82 -0
  142. package/src/tools/Chat/components/Sources.tsx +55 -0
  143. package/src/tools/Chat/components/StreamingIndicator.tsx +29 -0
  144. package/src/tools/Chat/components/ToolCalls.tsx +172 -0
  145. package/src/tools/Chat/components/index.ts +24 -0
  146. package/src/tools/Chat/config.ts +55 -0
  147. package/src/tools/Chat/context/ChatProvider.tsx +122 -0
  148. package/src/tools/Chat/context/index.ts +9 -0
  149. package/src/tools/Chat/core/audio/audioBus.ts +172 -0
  150. package/src/tools/Chat/core/audio/index.ts +8 -0
  151. package/src/tools/Chat/core/audio/preferences.ts +68 -0
  152. package/src/tools/Chat/core/audio/types.ts +49 -0
  153. package/src/tools/Chat/core/ids.ts +16 -0
  154. package/src/tools/Chat/core/index.ts +5 -0
  155. package/src/tools/Chat/core/markdown.ts +56 -0
  156. package/src/tools/Chat/core/payload-dispatch.ts +54 -0
  157. package/src/tools/Chat/core/persona.ts +35 -0
  158. package/src/tools/Chat/core/reducer.ts +335 -0
  159. package/src/tools/Chat/core/transport/http.ts +167 -0
  160. package/src/tools/Chat/core/transport/index.ts +13 -0
  161. package/src/tools/Chat/core/transport/mock.ts +134 -0
  162. package/src/tools/Chat/core/transport/sse.ts +116 -0
  163. package/src/tools/Chat/core/transport/types.ts +24 -0
  164. package/src/tools/Chat/hooks/index.ts +26 -0
  165. package/src/tools/Chat/hooks/useChat.ts +440 -0
  166. package/src/tools/Chat/hooks/useChatAudio.ts +191 -0
  167. package/src/tools/Chat/hooks/useChatComposer.ts +227 -0
  168. package/src/tools/Chat/hooks/useChatHistory.ts +59 -0
  169. package/src/tools/Chat/hooks/useChatLayout.ts +111 -0
  170. package/src/tools/Chat/hooks/useChatLightbox.ts +34 -0
  171. package/src/tools/Chat/hooks/useChatScroll.ts +132 -0
  172. package/src/tools/Chat/index.ts +158 -0
  173. package/src/tools/Chat/lazy.tsx +14 -0
  174. package/src/tools/Chat/types.ts +237 -0
  175. package/src/tools/Chat/utils/collectImageAttachments.ts +13 -0
  176. package/src/tools/JsonForm/JsonSchemaForm.tsx +32 -1
  177. package/src/tools/Map/README.md +384 -0
  178. package/dist/DocsLayout-CTJINVBM.mjs.map +0 -1
  179. package/dist/DocsLayout-XLDB6CJ2.cjs.map +0 -1
  180. package/dist/JsonSchemaForm-OSPUUUHM.cjs +0 -13
  181. package/dist/JsonSchemaForm-TSLX2GRO.mjs +0 -4
  182. package/dist/JsonTree-F27RMYSI.cjs +0 -11
  183. package/dist/JsonTree-QTJYSHCV.mjs +0 -5
  184. package/dist/Player-M3GC3VPE.mjs +0 -4
  185. package/dist/Player-ZL2X5LGG.cjs +0 -13
  186. package/dist/TreeRoot-A3J65L6F.mjs +0 -4
  187. package/dist/TreeRoot-DSK5JILT.cjs +0 -19
  188. package/dist/chunk-4IW7GZFQ.cjs.map +0 -1
  189. package/dist/chunk-62Y65TGK.mjs.map +0 -1
  190. package/dist/chunk-EXGXUK2N.mjs.map +0 -1
  191. package/dist/chunk-TKSFZHCG.cjs +0 -1597
  192. package/dist/chunk-TKSFZHCG.cjs.map +0 -1
  193. package/dist/components-5UXYNAKR.cjs +0 -22
  194. package/dist/components-CFXOEVPN.mjs +0 -5
  195. package/dist/components-WYEZL5TE.cjs +0 -26
  196. package/dist/components-ZAGG2PBO.mjs +0 -5
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/markdown/useCollapsibleContent.ts","../src/components/markdown/MarkdownMessage/plainText.ts","../src/tools/Mermaid/index.tsx","../src/tools/PrettyCode/index.tsx","../src/components/markdown/MarkdownMessage/sanitize.ts","../src/components/markdown/MarkdownMessage/CodeBlock.tsx","../src/components/markdown/MarkdownMessage/components.tsx","../src/components/markdown/MarkdownMessage/CollapseToggle.tsx","../src/components/markdown/MarkdownMessage/linkRules.ts","../src/components/markdown/MarkdownMessage/MarkdownMessage.tsx"],"names":["__name","useState","useMemo","useCallback","React","lazy","jsx","Suspense","LoadingFallback","jsxs","defaultSchema","useResolvedTheme","CopyButton","Fragment","ReactMarkdown","remarkGfm","remarkBreaks","remarkSmartypants","remarkEmoji","rehypeRaw","rehypeSanitize","rehypeExternalLinks"],"mappings":"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","file":"chunk-B5AWZOHJ.cjs","sourcesContent":["'use client';\n\nimport { useCallback, useMemo, useState } from 'react';\n\nexport interface UseCollapsibleContentOptions {\n /**\n * Maximum character length before collapsing\n * If both maxLength and maxLines are set, the stricter limit applies\n */\n maxLength?: number;\n /**\n * Maximum number of lines before collapsing\n * If both maxLength and maxLines are set, the stricter limit applies\n */\n maxLines?: number;\n /**\n * Start in expanded state (default: false - starts collapsed)\n */\n defaultExpanded?: boolean;\n}\n\nexport interface UseCollapsibleContentResult {\n /** Whether content is currently collapsed */\n isCollapsed: boolean;\n /** Toggle between collapsed/expanded state */\n toggleCollapsed: () => void;\n /** Set collapsed state directly */\n setCollapsed: (collapsed: boolean) => void;\n /** Content to display (truncated if collapsed, full if expanded) */\n displayContent: string;\n /** Whether the content exceeds limits and should be collapsible */\n shouldCollapse: boolean;\n /** Original content length */\n originalLength: number;\n /** Original line count */\n originalLineCount: number;\n}\n\n/**\n * Smart truncation that doesn't break words or markdown syntax\n */\nfunction smartTruncate(content: string, maxLength: number): string {\n if (content.length <= maxLength) {\n return content;\n }\n\n // Find a good break point (space, newline) near maxLength\n let breakPoint = maxLength;\n\n // Look backwards for a space or newline\n while (breakPoint > maxLength - 50 && breakPoint > 0) {\n const char = content[breakPoint];\n if (char === ' ' || char === '\\n' || char === '\\t') {\n break;\n }\n breakPoint--;\n }\n\n // If we couldn't find a good break point, just use maxLength\n if (breakPoint <= maxLength - 50) {\n breakPoint = maxLength;\n }\n\n let truncated = content.slice(0, breakPoint).trimEnd();\n\n // Fix unclosed markdown syntax\n truncated = fixUnclosedMarkdown(truncated);\n\n return truncated;\n}\n\n/**\n * Truncate by line count\n */\nfunction truncateByLines(content: string, maxLines: number): string {\n const lines = content.split('\\n');\n\n if (lines.length <= maxLines) {\n return content;\n }\n\n let truncated = lines.slice(0, maxLines).join('\\n').trimEnd();\n\n // Fix unclosed markdown syntax\n truncated = fixUnclosedMarkdown(truncated);\n\n return truncated;\n}\n\n/**\n * Fix unclosed markdown syntax to prevent rendering issues\n */\nfunction fixUnclosedMarkdown(content: string): string {\n let result = content;\n\n // Count occurrences of markdown markers\n const countOccurrences = (str: string, marker: string): number => {\n const escaped = marker.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n const matches = str.match(new RegExp(escaped, 'g'));\n return matches ? matches.length : 0;\n };\n\n // Fix unclosed bold (**) - must have even count\n const boldCount = countOccurrences(result, '**');\n if (boldCount % 2 !== 0) {\n result += '**';\n }\n\n // Fix unclosed italic (*) - but not **\n // Remove ** first for counting, then count single *\n const withoutBold = result.replace(/\\*\\*/g, '');\n const italicCount = countOccurrences(withoutBold, '*');\n if (italicCount % 2 !== 0) {\n result += '*';\n }\n\n // Fix unclosed inline code (`)\n const codeCount = countOccurrences(result, '`');\n // Ignore triple backticks for code blocks\n const tripleCount = countOccurrences(result, '```');\n const singleCodeCount = codeCount - (tripleCount * 3);\n if (singleCodeCount % 2 !== 0) {\n result += '`';\n }\n\n // Fix unclosed code blocks (```)\n if (tripleCount % 2 !== 0) {\n result += '\\n```';\n }\n\n // Fix unclosed strikethrough (~~)\n const strikeCount = countOccurrences(result, '~~');\n if (strikeCount % 2 !== 0) {\n result += '~~';\n }\n\n // Fix unclosed underline bold (__)\n const underlineBoldCount = countOccurrences(result, '__');\n if (underlineBoldCount % 2 !== 0) {\n result += '__';\n }\n\n // Fix unclosed underline italic (_) - but not __\n const withoutUnderlineBold = result.replace(/__/g, '');\n const underlineItalicCount = countOccurrences(withoutUnderlineBold, '_');\n if (underlineItalicCount % 2 !== 0) {\n result += '_';\n }\n\n return result;\n}\n\n/**\n * Hook for managing collapsible content with \"Read more...\" functionality\n *\n * @example\n * ```tsx\n * const { isCollapsed, toggleCollapsed, displayContent, shouldCollapse } = useCollapsibleContent(\n * longText,\n * { maxLength: 300, maxLines: 5 }\n * );\n *\n * return (\n * <div>\n * <Markdown content={displayContent} />\n * {shouldCollapse && (\n * <button onClick={toggleCollapsed}>\n * {isCollapsed ? 'Read more...' : 'Show less'}\n * </button>\n * )}\n * </div>\n * );\n * ```\n */\nexport function useCollapsibleContent(\n content: string,\n options: UseCollapsibleContentOptions = {}\n): UseCollapsibleContentResult {\n const { maxLength, maxLines, defaultExpanded = false } = options;\n\n const [isCollapsed, setIsCollapsed] = useState(!defaultExpanded);\n\n const originalLength = content.length;\n const originalLineCount = content.split('\\n').length;\n\n const { shouldCollapse, truncatedContent } = useMemo(() => {\n // If no limits set, don't collapse\n if (maxLength === undefined && maxLines === undefined) {\n return { shouldCollapse: false, truncatedContent: content };\n }\n\n let needsCollapse = false;\n let result = content;\n\n // Check line limit first (usually more restrictive for chat)\n if (maxLines !== undefined && originalLineCount > maxLines) {\n needsCollapse = true;\n result = truncateByLines(result, maxLines);\n }\n\n // Then check character limit\n if (maxLength !== undefined && result.length > maxLength) {\n needsCollapse = true;\n result = smartTruncate(result, maxLength);\n }\n\n return { shouldCollapse: needsCollapse, truncatedContent: result };\n }, [content, maxLength, maxLines, originalLineCount]);\n\n const displayContent = useMemo(() => {\n if (!shouldCollapse || !isCollapsed) {\n return content;\n }\n return truncatedContent;\n }, [content, truncatedContent, shouldCollapse, isCollapsed]);\n\n const toggleCollapsed = useCallback(() => {\n setIsCollapsed((prev) => !prev);\n }, []);\n\n const setCollapsed = useCallback((collapsed: boolean) => {\n setIsCollapsed(collapsed);\n }, []);\n\n return {\n isCollapsed,\n toggleCollapsed,\n setCollapsed,\n displayContent,\n shouldCollapse,\n originalLength,\n originalLineCount,\n };\n}\n\nexport default useCollapsibleContent;\n","import React from 'react';\n\n/** Recursively concatenate the text content of a React.ReactNode tree.\n * Used by the markdown renderers (e.g. `<pre>` extracting code) and\n * by consumers that need a plain-string label out of link children. */\nexport function extractTextFromChildren(children: React.ReactNode): string {\n if (typeof children === 'string') return children;\n if (typeof children === 'number') return String(children);\n if (React.isValidElement(children)) {\n const props = children.props as { children?: React.ReactNode };\n return extractTextFromChildren(props.children);\n }\n if (Array.isArray(children)) {\n return children.map(extractTextFromChildren).join('');\n }\n return '';\n}\n\n/**\n * Auto-detect whether `text` should bypass ReactMarkdown and render as\n * a flat `<div whitespace-pre-wrap>`. Used as the *fallback* when the\n * caller doesn't pass `plainText` explicitly to `MarkdownMessage`.\n *\n * The signal we trust: short, single-paragraph, no markdown markers.\n * Anything longer / multi-paragraph / structurally suggestive falls\n * through to the markdown pipeline — false negatives there are cheap\n * (markdown renders prose correctly), false positives in the prose\n * branch surface as escaped `*` / `#` / etc. so we err markdown-ward.\n *\n * Thresholds were picked from chat-bubble UX research (see ChatGPT,\n * Claude.ai, WhatsApp): roughly \"would a person have written this in\n * one keystroke without thinking about formatting?\". If you find them\n * too tight or too loose, tune here — every consumer goes through\n * `MarkdownMessage` so the change is universal.\n */\nexport function looksLikePlainProse(text: string): boolean {\n const trimmed = text.trim();\n // Empty / whitespace-only — render as plain (cheap path, nothing to parse).\n if (trimmed.length === 0) return true;\n // Long enough that it's likely a structured assistant reply.\n if (trimmed.length > 500) return false;\n // Paragraph break → almost certainly markdown territory.\n if (/\\n\\s*\\n/.test(trimmed)) return false;\n // Many single-line breaks → probably a list or stanza.\n const newlineCount = (trimmed.match(/\\n/g) || []).length;\n if (newlineCount > 4) return false;\n // Any markdown marker → defer to ReactMarkdown.\n if (hasMarkdownSyntax(trimmed)) return false;\n return true;\n}\n\n/** Affordance test: does this string look like markdown? Used to skip\n * the (heavier) ReactMarkdown pipeline when the content is pure\n * prose. NOT a validator — false negatives are fine; false positives\n * cost a render but render correctly. */\nexport function hasMarkdownSyntax(text: string): boolean {\n // Newlines after trim → render as markdown so paragraphs work.\n if (text.trim().includes('\\n')) return true;\n\n // Inline HTML tags (`<br>`, `<b>`, `<code>`, …) — common in OpenAPI\n // descriptions. Without this branch the fast path would escape them\n // and the user sees literal angle brackets.\n if (/<\\/?[a-zA-Z][a-zA-Z0-9-]*(\\s[^>]*)?\\/?>/.test(text)) return true;\n\n const patterns = [\n /^#{1,6}\\s/m, // Headers\n /\\*\\*[^*]+\\*\\*/, // Bold\n /\\*[^*]+\\*/, // Italic\n /__[^_]+__/, // Bold (underscore)\n /_[^_]+_/, // Italic (underscore)\n /\\[.+\\]\\(.+\\)/, // Links\n /!\\[.*\\]\\(.+\\)/, // Images\n /```[\\s\\S]*```/, // Code blocks\n /`[^`]+`/, // Inline code\n /^\\s*[-*+]\\s/m, // Unordered lists\n /^\\s*\\d+\\.\\s/m, // Ordered lists\n /^\\s*>/m, // Blockquotes\n /\\|.+\\|/, // Tables\n /^---+$/m, // Horizontal rules\n /~~[^~]+~~/, // Strikethrough\n ];\n return patterns.some((p) => p.test(text));\n}\n","/**\n * Mermaid Component - Dynamic Import Wrapper\n *\n * Lazy loads the heavy Mermaid library (~800KB) only when needed.\n * This significantly reduces the initial bundle size.\n */\n\n'use client';\n\nimport React, { lazy, Suspense } from 'react';\n\n// Lazy load the client component\nconst MermaidClient = lazy(() => import('./Mermaid.client'));\n\n// Loading fallback component\nconst LoadingFallback = () => (\n <div className=\"flex justify-center items-center min-h-[100px]\">\n <div className=\"animate-spin rounded-full h-6 w-6 border-b-2 border-primary\" />\n </div>\n);\n\nexport interface MermaidProps {\n chart: string;\n className?: string;\n isCompact?: boolean;\n /** Enable click-to-fullscreen functionality (default: true) */\n fullscreen?: boolean;\n /**\n * Enable the FloatingToolbar's \"click to scroll\" lock overlay.\n * Defaults to `false` — Mermaid diagrams don't scroll internally,\n * so the lock overlay just steals page wheel events. See the\n * Mermaid.client implementation for the full rationale.\n */\n scrollIsolation?: boolean;\n}\n\nconst Mermaid: React.FC<MermaidProps> = (props) => {\n return (\n <Suspense fallback={<LoadingFallback />}>\n <MermaidClient {...props} />\n </Suspense>\n );\n};\n\nexport default Mermaid;\n\n// Re-export builders for declarative diagram construction\nexport {\n // Core\n DiagramStore,\n sanitizeLabel,\n toNodeId,\n // Theme hooks\n useThemePalette,\n useStylePresets,\n useBoxColors,\n // FlowDiagram\n FlowDiagram,\n STYLE_PRESETS,\n // SequenceDiagram\n SequenceDiagram,\n // JourneyDiagram\n JourneyDiagram,\n} from './builders';\n\nexport type {\n // Core types\n DiagramStoreOptions,\n FlowDirection,\n NodeShape,\n ParticipantType,\n TaskScore,\n // Theme types\n ThemePalette,\n StyleColors,\n StylePresets,\n BoxColors,\n // FlowDiagram types\n FlowDiagramOptions,\n FlowDiagramBuilder,\n NodeBuilder,\n EdgeBuilder,\n StyleBuilder,\n // SequenceDiagram types\n ParticipantsObject,\n SequenceDiagramOptions,\n SequenceDiagramBuilder,\n // JourneyDiagram types\n JourneyDiagramOptions,\n JourneyDiagramBuilder,\n SectionBuilder,\n} from './builders';\n","/**\n * PrettyCode Component - Dynamic Import Wrapper\n *\n * Lazy loads the heavy Prism library (~500KB) only when needed.\n * This significantly reduces the initial bundle size.\n */\n\n'use client';\n\nimport React, { lazy, Suspense } from 'react';\nimport type { Language } from 'prism-react-renderer';\n\n// Lazy load the client component\nconst PrettyCodeClient = lazy(() => import('./PrettyCode.client'));\n\n// Loading fallback component\nconst LoadingFallback = () => (\n <div className=\"relative rounded-sm border border-border overflow-hidden bg-muted dark:bg-zinc-900\">\n <div className=\"p-4\">\n <div className=\"flex items-center gap-2\">\n <div className=\"animate-pulse h-4 w-4 rounded-full bg-muted-foreground/20\"></div>\n <span className=\"text-xs text-muted-foreground\">Loading code...</span>\n </div>\n </div>\n </div>\n);\n\nexport interface PrettyCodeProps {\n data: string | object;\n language: Language;\n className?: string;\n mode?: 'dark' | 'light';\n inline?: boolean;\n customBg?: string;\n isCompact?: boolean;\n /** Block scroll capture until user clicks (default: true) */\n scrollIsolation?: boolean;\n /**\n * Line count above which the block starts scrolling instead of growing.\n * ``undefined`` (default) = no cap, block grows to fit. Set e.g. ``50``\n * to inline short snippets and cap long ones.\n */\n maxLines?: number;\n /**\n * Visual variant:\n * - ``\"card\"`` (default) — full chrome: border, background, hover\n * toolbar (Copy + language tag), optional internal scroll. Suits\n * a standalone code block in documentation prose where the block\n * is its own surface.\n * - ``\"plain\"`` — chrome-less: no border, no background, no toolbar,\n * no internal scroll. Grows to fit its content. Use when\n * embedding inside another scroll container (e.g. the Response\n * panel) — avoids double-scroll surfaces and lets the parent\n * manage copy/language affordances.\n */\n variant?: 'card' | 'plain';\n}\n\nconst PrettyCode: React.FC<PrettyCodeProps> = (props) => {\n return (\n <Suspense fallback={<LoadingFallback />}>\n <PrettyCodeClient {...props} />\n </Suspense>\n );\n};\n\nexport default PrettyCode;\nexport type { Language };\n","import { defaultSchema } from 'rehype-sanitize';\n\n/** Sanitize schema. Typed loosely (`Record<string, unknown>`) so this\n * package's published types don't try to re-export\n * `hast-util-sanitize`'s internal Schema type — that breaks consumers\n * whose TS doesn't have the transitive dependency in scope. */\nexport type SanitizeSchema = Record<string, unknown>;\n\n// Allow-list HTML that OpenAPI descriptions commonly use — we want\n// `<b>`/`<code>`/`<br>` to render, not to appear as literal text.\n// Built on top of rehype-sanitize's default schema so we keep the\n// XSS protection (no `<script>`, no `on*` handlers, no `javascript:`\n// URLs); we just extend the tag allow-list with safe presentational\n// elements that are in wide use in API docs.\nexport const HTML_SCHEMA_BASE: SanitizeSchema = {\n ...defaultSchema,\n tagNames: [\n ...(defaultSchema.tagNames ?? []),\n 'br',\n 'b',\n 'i',\n 'u',\n 's',\n 'sub',\n 'sup',\n 'small',\n 'mark',\n 'kbd',\n 'code',\n 'pre',\n 'details',\n 'summary',\n ],\n};\n\n/** Build a sanitize schema with extra href-URL protocols allowed on\n * `<a>`. Default whitelists `http(s)/mailto/xmpp/irc(s)` — pass extras\n * here (e.g. `'cmdop'`, `'obsidian'`) to opt them in. */\nexport function buildSchema(extraProtocols: readonly string[] | undefined): SanitizeSchema {\n if (!extraProtocols || extraProtocols.length === 0) return HTML_SCHEMA_BASE;\n const baseProtocols =\n (HTML_SCHEMA_BASE as { protocols?: Record<string, string[]> }).protocols\n ?? (defaultSchema as { protocols?: Record<string, string[]> }).protocols\n ?? {};\n const baseHref = baseProtocols.href ?? ['http', 'https', 'mailto', 'xmpp', 'irc', 'ircs'];\n return {\n ...HTML_SCHEMA_BASE,\n protocols: {\n ...baseProtocols,\n href: [...baseHref, ...extraProtocols],\n },\n };\n}\n\n/** Build a `urlTransform` for ReactMarkdown that lets opted-in schemes\n * pass through verbatim. react-markdown's default strips `href` for\n * unrecognised schemes BEFORE sanitize runs — even with our extended\n * sanitize schema, custom schemes would arrive with `href=\"\"`. */\nexport function buildUrlTransform(extraProtocols: readonly string[] | undefined) {\n if (!extraProtocols || extraProtocols.length === 0) return undefined;\n const lower = extraProtocols.map((p) => p.toLowerCase() + ':');\n return (url: string): string => {\n const u = url.trim().toLowerCase();\n for (const p of lower) {\n if (u.startsWith(p)) return url;\n }\n // Fallback: replicate react-markdown's defaults for safe schemes\n // and relative paths. Anything else → empty string so the\n // anchor's href is dropped (XSS guard).\n if (\n /^(https?:|mailto:|tel:|xmpp:|irc:|ircs:|#|\\/|\\.\\/|\\.\\.\\/|\\?)/i.test(u)\n || /^[a-z0-9._~!$&'()*+,;=:@%-]+$/i.test(u)\n ) {\n return url;\n }\n return '';\n };\n}\n","import React from 'react';\nimport { CopyButton } from '@djangocfg/ui-core/components';\nimport { useResolvedTheme } from '@djangocfg/ui-core/hooks';\nimport PrettyCode from '../../../tools/PrettyCode';\n\ninterface CodeBlockProps {\n code: string;\n language: string;\n isUser: boolean;\n isCompact?: boolean;\n}\n\n/**\n * Markdown code block. Renders <PrettyCode> directly — its own\n * FloatingToolbar already carries the language tag and Copy action.\n *\n * Earlier versions stacked an extra <CopyButton> here too, which\n * surfaced as two copy buttons on every code fence. The fallback\n * branch below still ships its own button because the plain <pre>\n * has no toolbar of its own.\n */\nexport const CodeBlock: React.FC<CodeBlockProps> = ({ code, language }) => {\n const theme = useResolvedTheme();\n\n // Chat fences are always rendered in PrettyCode's compact mode:\n // 12px font, tighter padding, line-height 1.4. A fenced block in\n // a chat bubble shouldn't outweigh two paragraphs of body text —\n // the standalone PrettyCode story keeps the larger default for\n // docs/diff viewers.\n return (\n <div className=\"my-2\">\n <PrettyCode\n data={code}\n language={language}\n className=\"text-xs\"\n customBg=\"bg-code\"\n mode={theme}\n isCompact\n // Disable click-to-scroll-isolation in chat markdown: code\n // fences here are part of an assistant reply, not a docs\n // viewer. Forcing the user to click into a small block to\n // scroll past it interrupts the reading flow. The standalone\n // PrettyCode use cases (docs, diff viewers, long code panes)\n // keep the default `true`.\n scrollIsolation={false}\n />\n </div>\n );\n};\n\n/** Simple `<pre>` fallback used when CodeBlock throws (lazy module\n * failure, missing PrettyCode peer, etc). */\nexport const CodeBlockFallback: React.FC<CodeBlockProps> = ({ code, isUser }) => {\n // See CodeBlock above for the spirit of this layout — palette\n // pre-computed before render.\n const copyHoverClass = isUser\n ? 'hover:bg-white/20 text-white'\n : 'hover:bg-muted-foreground/20 text-muted-foreground hover:text-foreground';\n const copyButtonClass =\n `absolute top-2 right-2 z-10 opacity-0 group-hover:opacity-100 transition-opacity ` +\n `h-8 w-8 ${copyHoverClass}`;\n\n return (\n <div className=\"relative group my-3\">\n <CopyButton\n value={code}\n variant=\"ghost\"\n className={copyButtonClass}\n title=\"Copy code\"\n />\n <pre className=\"p-3 rounded text-xs font-mono overflow-x-auto bg-code text-code-foreground border border-code-border\">\n <code>{code}</code>\n </pre>\n </div>\n );\n};\n","import React from 'react';\nimport type { Components } from 'react-markdown';\nimport Mermaid from '../../../tools/Mermaid';\nimport { CodeBlock, CodeBlockFallback } from './CodeBlock';\nimport { extractTextFromChildren } from './plainText';\n\n/**\n * Build the chat-tuned markdown component map.\n *\n * Base text size: text-sm (14px) for normal, text-xs (12px) for compact.\n * Heading sizes are scaled down a notch from the prose defaults so\n * inline-in-chat headings don't dominate.\n */\nexport function createMarkdownComponents(\n isUser: boolean = false,\n isCompact: boolean = false,\n): Components {\n const textSize = isCompact ? 'text-xs' : 'text-sm';\n const headingBase = isCompact ? 'text-sm' : 'text-base';\n const headingSm = isCompact ? 'text-xs' : 'text-sm';\n\n return {\n h1: ({ children }) => (\n <h1 className={`${headingBase} font-semibold mb-2 mt-3 first:mt-0`}>{children}</h1>\n ),\n h2: ({ children }) => (\n <h2 className={`${headingSm} font-semibold mb-2 mt-3 first:mt-0`}>{children}</h2>\n ),\n h3: ({ children }) => (\n <h3 className={`${headingSm} font-semibold mb-1 mt-2 first:mt-0`}>{children}</h3>\n ),\n h4: ({ children }) => (\n <h4 className={`${headingSm} font-semibold mb-1 mt-2 first:mt-0`}>{children}</h4>\n ),\n h5: ({ children }) => (\n <h5 className={`${headingSm} font-medium mb-1 mt-2 first:mt-0`}>{children}</h5>\n ),\n h6: ({ children }) => (\n <h6 className={`${headingSm} font-medium mb-1 mt-2 first:mt-0`}>{children}</h6>\n ),\n\n p: ({ children }) => (\n <p className={`${textSize} mb-4 last:mb-0 leading-relaxed break-words`}>{children}</p>\n ),\n\n ul: ({ children }) => (\n <ul className={`list-disc list-inside mb-2 space-y-1 ${textSize}`}>{children}</ul>\n ),\n ol: ({ children }) => (\n <ol className={`list-decimal list-inside mb-2 space-y-1 ${textSize}`}>{children}</ol>\n ),\n li: ({ children }) => <li className=\"break-words\">{children}</li>,\n\n // `target` / `rel` for external links are NOT set here — the\n // rehype-external-links plugin tags them on the rehype side, so\n // every `<a>` that sanitize let through gets the same security\n // treatment regardless of which renderer (default vs linkRules\n // override) emitted it. Doing it twice here would just duplicate\n // attributes; doing it only here would miss the linkRules path.\n a: ({ href, children, ...rest }) => (\n <a\n {...rest}\n href={href}\n className={`${textSize} ${\n isUser\n ? 'text-white/90 underline hover:text-white'\n : 'text-primary underline hover:text-primary/80'\n } transition-colors break-all`}\n >\n {children}\n </a>\n ),\n\n pre: ({ children }) => {\n let codeContent = '';\n let language = 'plaintext';\n\n if (React.isValidElement(children)) {\n const child = children;\n if (\n child.type === 'code'\n || (typeof child.type === 'function' && child.type.name === 'code')\n ) {\n const codeProps = child.props as {\n className?: string;\n children?: React.ReactNode;\n };\n const rawClassName = codeProps.className;\n language = rawClassName?.replace(/language-/, '').trim() || 'plaintext';\n codeContent = extractTextFromChildren(codeProps.children).trim();\n } else {\n codeContent = extractTextFromChildren(children).trim();\n }\n } else {\n codeContent = extractTextFromChildren(children).trim();\n }\n\n if (!codeContent) {\n return (\n <div className=\"my-3 p-3 bg-muted rounded text-sm text-muted-foreground\">\n No content available\n </div>\n );\n }\n\n if (language === 'mermaid') {\n // Inline render fits the bubble width; the Mermaid component\n // owns its own click-to-fullscreen modal which sizes against\n // the viewport, so we don't cap it here. A previous version\n // hardcoded `max-w-[600px]` and that constraint leaked into\n // the fullscreen modal too — diagram rendered at 600px in the\n // middle of an empty viewport.\n return (\n <div className=\"my-3 w-full\">\n <Mermaid chart={codeContent} isCompact={isCompact} />\n </div>\n );\n }\n\n try {\n return <CodeBlock code={codeContent} language={language} isUser={isUser} isCompact={isCompact} />;\n } catch (error) {\n // eslint-disable-next-line no-console\n console.warn('CodeBlock failed, using fallback:', error);\n return <CodeBlockFallback code={codeContent} language={language} isUser={isUser} isCompact={isCompact} />;\n }\n },\n\n code: ({ children, className }) => {\n // Inside <pre>: let pre handle styling.\n if (className?.includes('language-')) {\n return <code className={className}>{children}</code>;\n }\n // Inline `<code>` uses the design system's `--code-inline`\n // token. One semantic chip surface across both themes; on a\n // user bubble we still palette-switch with `text-primary-\n // foreground` because the inline chip blends INTO the bubble\n // — there's no panel boundary like a fence has. We trade a\n // perfect \"code surface\" tone here for legible body inheritance,\n // matching ChatGPT's behaviour in coloured user bubbles.\n const inlineCodeClass = isUser\n ? 'bg-primary-foreground/15 text-primary-foreground'\n : 'bg-code-inline text-code-inline-foreground';\n return (\n <code className={`px-1 py-0.5 rounded font-mono text-[0.875em] ${inlineCodeClass} break-all`}>\n {extractTextFromChildren(children)}\n </code>\n );\n },\n\n // Modern chat convention drops italic on blockquotes — italic +\n // tight bubble = hard to read. Border-left at 2px (4px reads\n // heavy in a 320–480px bubble). On the saturated user bubble we\n // use a primary-foreground tint; on the assistant bubble we use\n // the muted-foreground role for de-emphasis.\n blockquote: ({ children }) => {\n const cls = isUser\n ? 'border-primary-foreground/40 text-primary-foreground/80'\n : 'border-border text-muted-foreground';\n return (\n <blockquote className={`${textSize} border-l-2 pl-3 my-3 break-words ${cls}`}>\n {children}\n </blockquote>\n );\n },\n\n // Tables: outer wrapper handles overflow, inner `<table>`\n // inherits the chat-density text size. Borders / header use\n // semantic tokens — `border-code-border` for the assistant\n // (matches the code-fence panel for visual cohesion when both\n // appear in the same reply); primary-foreground/N for the user\n // bubble so lines read against the saturated `bg-primary`.\n table: ({ children }) => (\n <div className=\"overflow-x-auto my-3\">\n <table className={`min-w-full ${textSize} border-collapse`}>{children}</table>\n </div>\n ),\n thead: ({ children }) => (\n <thead className={isUser ? 'bg-primary-foreground/10' : 'bg-muted/40'}>\n {children}\n </thead>\n ),\n tbody: ({ children }) => <tbody>{children}</tbody>,\n tr: ({ children }) => (\n <tr className={isUser ? 'border-b border-primary-foreground/15' : 'border-b border-border'}>\n {children}\n </tr>\n ),\n th: ({ children }) => {\n const borderCls = isUser ? 'border-primary-foreground/25' : 'border-border';\n return (\n <th className={`px-2 py-1.5 text-left font-semibold border-b ${borderCls} break-words`}>\n {children}\n </th>\n );\n },\n td: ({ children }) => <td className=\"px-2 py-1.5 break-words\">{children}</td>,\n\n // Soft separator. ChatGPT / Slack / Linear strip the visible\n // line, Claude.ai keeps a hairline. We follow Claude — present\n // but quiet. Palette switches by role so the hairline reads on\n // both surfaces.\n hr: () => (\n <hr\n className={`my-4 border-0 h-px ${\n isUser ? 'bg-primary-foreground/20' : 'bg-border'\n }`}\n />\n ),\n\n strong: ({ children }) => <strong className=\"font-semibold\">{children}</strong>,\n em: ({ children }) => <em className=\"italic\">{children}</em>,\n };\n}\n","import React from 'react';\n\ninterface CollapseToggleProps {\n isCollapsed: boolean;\n onClick: () => void;\n readMoreLabel: string;\n showLessLabel: string;\n isUser: boolean;\n isCompact: boolean;\n}\n\n/** \"Read more...\" / \"Show less\" button. */\nexport const CollapseToggle: React.FC<CollapseToggleProps> = ({\n isCollapsed,\n onClick,\n readMoreLabel,\n showLessLabel,\n isUser,\n isCompact,\n}) => {\n const textSize = isCompact ? 'text-xs' : 'text-sm';\n return (\n <button\n type=\"button\"\n onClick={onClick}\n className={`\n ${textSize} font-medium cursor-pointer\n transition-colors duration-200\n ${isUser ? 'text-white/80 hover:text-white' : 'text-primary hover:text-primary/80'}\n inline-flex items-center gap-1\n mt-1\n `}\n >\n {isCollapsed ? (\n <>\n {readMoreLabel}\n <Chevron direction=\"down\" />\n </>\n ) : (\n <>\n {showLessLabel}\n <Chevron direction=\"up\" />\n </>\n )}\n </button>\n );\n};\n\nfunction Chevron({ direction }: { direction: 'up' | 'down' }) {\n return (\n <svg className=\"w-3 h-3\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n strokeWidth={2}\n d={direction === 'down' ? 'M19 9l-7 7-7-7' : 'M5 15l7-7 7 7'}\n />\n </svg>\n );\n}\n","import React from 'react';\nimport type { ComponentProps, ComponentType } from 'react';\nimport type { Components } from 'react-markdown';\nimport type { LinkRule } from './types';\n\n// react-markdown's `Components['a']` is `keyof IntrinsicElements | ComponentType<…>`,\n// so it isn't necessarily a function we can directly call. This is the\n// component-shaped variant — what callers practically pass when they\n// override `a`.\ntype AComponent = ComponentType<ComponentProps<'a'>>;\n\n/** Run every rule's `preprocess` hook in order. Errors in a single\n * rule are logged and skipped — the other rules still run. */\nexport function applyPreprocess(\n source: string,\n rules: readonly LinkRule[] | undefined,\n): string {\n if (!rules || rules.length === 0) return source;\n let s = source;\n for (const rule of rules) {\n if (!rule.preprocess) continue;\n try {\n s = rule.preprocess(s);\n } catch (err) {\n // eslint-disable-next-line no-console\n console.warn(\n `[MarkdownMessage] linkRule \"${rule.name ?? '(anonymous)'}\" preprocess threw; skipping`,\n err,\n );\n }\n }\n return s;\n}\n\n/** Union of `extraHrefProtocols` and any `protocols` declared by rules. */\nexport function collectProtocols(\n extraHrefProtocols: readonly string[] | undefined,\n rules: readonly LinkRule[] | undefined,\n): readonly string[] | undefined {\n const set = new Set<string>();\n if (extraHrefProtocols) for (const p of extraHrefProtocols) set.add(p);\n if (rules) {\n for (const r of rules) {\n if (r.protocols) for (const p of r.protocols) set.add(p);\n }\n }\n return set.size === 0 ? undefined : Array.from(set);\n}\n\n/** Build a custom `a` renderer that dispatches to the first matching\n * rule, falling through to `callerA` (or the built-in chat anchor)\n * for anything no rule claims. */\nexport function buildLinkRulesComponent(\n rules: readonly LinkRule[],\n isUser: boolean,\n callerA: NonNullable<Components['a']> | undefined,\n): NonNullable<Components['a']> {\n const Renderer: AComponent = (props) => {\n const { href, children } = props;\n if (typeof href === 'string') {\n for (const rule of rules) {\n if (rule.match(href)) {\n return React.createElement(\n React.Fragment,\n null,\n rule.render({ href, children, isUser }),\n );\n }\n }\n }\n // Defer to the caller's `a` override if any. We only call it when\n // it's a function/component — string-form intrinsic overrides\n // (`'a' | 'span' | …`) aren't a thing react-markdown supports for\n // tag overrides, but the type still admits the union, so guard.\n if (callerA && typeof callerA === 'function') {\n const Caller = callerA as AComponent;\n return React.createElement(Caller, props);\n }\n // Fall through to the built-in anchor by rendering a plain `<a>`.\n return React.createElement('a', props, children);\n };\n return Renderer as NonNullable<Components['a']>;\n}\n","'use client';\n\nimport React from 'react';\nimport ReactMarkdown from 'react-markdown';\nimport rehypeExternalLinks from 'rehype-external-links';\nimport rehypeRaw from 'rehype-raw';\nimport rehypeSanitize from 'rehype-sanitize';\nimport remarkBreaks from 'remark-breaks';\nimport remarkEmoji from 'remark-emoji';\nimport remarkGfm from 'remark-gfm';\nimport remarkSmartypants from 'remark-smartypants';\nimport type { Components } from 'react-markdown';\n\nimport { useCollapsibleContent } from '../useCollapsibleContent';\nimport type { MarkdownMessageProps } from './types';\nimport { buildSchema, buildUrlTransform } from './sanitize';\nimport { looksLikePlainProse } from './plainText';\nimport { createMarkdownComponents } from './components';\nimport { CollapseToggle } from './CollapseToggle';\nimport { applyPreprocess, buildLinkRulesComponent, collectProtocols } from './linkRules';\n\n/**\n * MarkdownMessage — chat-tuned markdown renderer.\n *\n * Features:\n * - GitHub Flavored Markdown (GFM) via remark-gfm\n * - Syntax-highlighted code blocks with Copy button\n * - Mermaid diagram rendering (` ```mermaid ` fence)\n * - Tables, lists, blockquotes scaled for chat density\n * - User vs assistant styling modes (`isUser`)\n * - Plain-text fast path: skips ReactMarkdown when content has no\n * markdown syntax (cheaper render, preserves newlines via CSS)\n * - Optional collapsible \"Read more...\" for long messages\n *\n * Custom URL schemes (chat mentions, deep-links, custom file viewers)\n * are best handled with the declarative `linkRules` prop — see the\n * type definition in `./types.ts` and the storybook for examples.\n *\n * @example\n * ```tsx\n * <MarkdownMessage content=\"# Hello\\n\\nThis is **bold** text.\" />\n *\n * // User message styling\n * <MarkdownMessage content=\"Some content\" isUser />\n *\n * // Custom URL scheme via linkRules\n * <MarkdownMessage\n * content=\"Talk to [Vps-audi](cmdop://machine/abc-123)\"\n * linkRules={[machineMentionRule]}\n * />\n * ```\n */\nexport const MarkdownMessage: React.FC<MarkdownMessageProps> = ({\n content,\n className = '',\n isUser = false,\n isCompact = false,\n plainText,\n customComponents,\n extraHrefProtocols,\n linkRules,\n collapsible = false,\n maxLength,\n maxLines,\n readMoreLabel = 'Read more...',\n showLessLabel = 'Show less',\n defaultExpanded = false,\n onCollapseChange,\n}) => {\n // Pre-process content through any rules that requested it. Done\n // before trim so a rule can rewrite multi-line shapes too.\n const preprocessed = React.useMemo(\n () => applyPreprocess(content, linkRules),\n [content, linkRules],\n );\n\n // Union of `extraHrefProtocols` and any `protocols` declared by rules.\n const effectiveProtocols = React.useMemo(\n () => collectProtocols(extraHrefProtocols, linkRules),\n [extraHrefProtocols, linkRules],\n );\n\n // Effective custom components: merge linkRules' synthesized `a`\n // renderer with any caller-provided `customComponents`. linkRules\n // wins when both target the same href (rule is the more specific\n // declarative claim by design).\n const effectiveCustomComponents = React.useMemo<Partial<Components> | undefined>(() => {\n if (!linkRules || linkRules.length === 0) return customComponents;\n const callerA = customComponents?.a;\n const aRenderer = buildLinkRulesComponent(linkRules, isUser, callerA);\n return { ...(customComponents ?? {}), a: aRenderer };\n }, [customComponents, linkRules, isUser]);\n\n const trimmedContent = preprocessed.trim();\n\n // Collapsible content logic — defaults kick in only when enabled.\n const collapsibleOptions = React.useMemo(() => {\n if (!collapsible) return {};\n return {\n maxLength: maxLength ?? 1000,\n maxLines: maxLines ?? 10,\n defaultExpanded,\n };\n }, [collapsible, maxLength, maxLines, defaultExpanded]);\n\n const { isCollapsed, toggleCollapsed, displayContent, shouldCollapse } =\n useCollapsibleContent(trimmedContent, collapsible ? collapsibleOptions : {});\n\n React.useEffect(() => {\n if (collapsible && shouldCollapse && onCollapseChange) {\n onCollapseChange(isCollapsed);\n }\n }, [isCollapsed, collapsible, shouldCollapse, onCollapseChange]);\n\n const components = React.useMemo(() => {\n const base = createMarkdownComponents(isUser, isCompact);\n return effectiveCustomComponents ? { ...base, ...effectiveCustomComponents } : base;\n }, [isUser, isCompact, effectiveCustomComponents]);\n\n const schema = React.useMemo(() => buildSchema(effectiveProtocols), [effectiveProtocols]);\n const urlTransform = React.useMemo(\n () => buildUrlTransform(effectiveProtocols),\n [effectiveProtocols],\n );\n\n const textSizeClass = isCompact ? 'text-xs' : 'text-sm';\n const proseClass = isCompact ? 'prose-xs' : 'prose-sm';\n\n // Resolve plain-vs-markdown branch:\n // 1. Caller-passed `plainText` wins outright (explicit beats clever).\n // 2. Caller-supplied non-`a` customComponents force the markdown\n // pipeline — those overrides only matter on real markdown nodes.\n // 3. Otherwise auto-detect via `looksLikePlainProse` (short,\n // single-paragraph, no markdown markers). This is the chat-bubble\n // WhatsApp/Telegram heuristic — \"would the human have written\n // this in one keystroke?\".\n const customComponentsBeyondLinks = React.useMemo(() => {\n if (!customComponents) return false;\n return Object.keys(customComponents).some((k) => k !== 'a');\n }, [customComponents]);\n const isPlainText = plainText !== undefined\n ? plainText\n : !customComponentsBeyondLinks && looksLikePlainProse(displayContent);\n\n if (isPlainText) {\n // <div> + whitespace-pre-wrap: respects newlines AND collapses\n // double spaces, which is what users mean when they hit Enter\n // twice. <span> would break flow inside a flex bubble.\n //\n // leading-snug (1.375), not leading-relaxed (1.625) — `pre-wrap`\n // makes every `\\n` a hard line break, so the relaxed leading\n // turns multi-line user messages into airy ladders. snug matches\n // WhatsApp/Telegram bubble density.\n return (\n <div\n className={`${textSizeClass} font-normal antialiased leading-snug break-words whitespace-pre-wrap ${className}`}\n >\n {displayContent}\n {collapsible && shouldCollapse && (\n <>\n {isCollapsed && '... '}\n <CollapseToggle\n isCollapsed={isCollapsed}\n onClick={toggleCollapsed}\n readMoreLabel={readMoreLabel}\n showLessLabel={showLessLabel}\n isUser={isUser}\n isCompact={isCompact}\n />\n </>\n )}\n </div>\n );\n }\n\n return (\n <div className={className}>\n <div\n className={`\n prose ${proseClass} max-w-none break-words overflow-hidden ${textSizeClass} font-normal antialiased\n ${isUser ? 'prose-invert' : 'dark:prose-invert'}\n [&>*]:leading-relaxed\n [&>*:first-child]:mt-0 [&>*:last-child]:mb-0\n [&_p]:my-2\n [&_ul]:my-2 [&_ol]:my-2 [&_ul]:pl-5 [&_ol]:pl-5\n [&_li]:my-1 [&_li>p]:my-0\n [&_h1]:mt-4 [&_h1]:mb-2 [&_h1]:text-base [&_h1]:font-semibold\n [&_h2]:mt-3.5 [&_h2]:mb-1.5 [&_h2]:text-[15px] [&_h2]:font-semibold\n [&_h3]:mt-3 [&_h3]:mb-1 [&_h3]:text-sm [&_h3]:font-medium\n [&_h4]:mt-3 [&_h4]:mb-1 [&_h4]:text-sm [&_h4]:font-medium\n `}\n style={{\n // Inherit colors from parent — fixes issues with external\n // CSS variables overriding prose tokens.\n '--tw-prose-body': 'inherit',\n '--tw-prose-headings': 'inherit',\n '--tw-prose-bold': 'inherit',\n '--tw-prose-links': 'inherit',\n color: 'inherit',\n } as React.CSSProperties}\n >\n <ReactMarkdown\n // Remark plugin order is load-bearing:\n // 1. `remark-gfm` — tables, strikethrough, autolinks, task lists.\n // 2. `remark-breaks` — chat convention: single `\\n` → `<br>`\n // (ChatGPT / Slack / Discord / Linear). CommonMark's\n // default would collapse those into one paragraph and\n // LLM punchlines / poems / dialogue would render as a\n // run-on line. Goes BEFORE smartypants so quotes that\n // land at line breaks still get the curly treatment.\n // 3. `remark-smartypants` — typographic substitutions:\n // \"...\" → …, -- → —, \"x\" → \"x\". Cheap \"humanized\"\n // polish à la Medium / Substack.\n // 4. `remark-emoji` — `:smile:` → 😄 (GitHub / Linear\n // shortcode style). Leaves Unicode emoji untouched.\n remarkPlugins={[remarkGfm, remarkBreaks, remarkSmartypants, remarkEmoji]}\n // rehype-raw parses inline HTML in the source; rehype-sanitize\n // (with our extended schema) runs after to keep XSS guards\n // (no scripts, no on* handlers, no javascript: urls).\n // rehype-external-links runs LAST so it tags every <a> that\n // sanitize let through — externals get target=_blank +\n // rel=noopener noreferrer in one pass, instead of every\n // custom `a` renderer reimplementing the rule.\n rehypePlugins={[\n rehypeRaw,\n [rehypeSanitize, schema],\n [rehypeExternalLinks, { target: '_blank', rel: ['noopener', 'noreferrer'] }],\n ]}\n components={components}\n // urlTransform runs in remark-rehype before sanitize. Without\n // overriding it, react-markdown's default strips `href` for\n // unknown schemes — making our extended sanitize whitelist\n // moot. Only set when the caller opted into extra protocols.\n urlTransform={urlTransform}\n >\n {displayContent}\n </ReactMarkdown>\n </div>\n {collapsible && shouldCollapse && (\n <CollapseToggle\n isCollapsed={isCollapsed}\n onClick={toggleCollapsed}\n readMoreLabel={readMoreLabel}\n showLessLabel={showLessLabel}\n isUser={isUser}\n isCompact={isCompact}\n />\n )}\n </div>\n );\n};\n\nexport default MarkdownMessage;\n"]}
@@ -1,4 +1,4 @@
1
- import { __name } from './chunk-CGILA3WO.mjs';
1
+ import { __name } from './chunk-N2XQF2OL.mjs';
2
2
  import { cn } from '@djangocfg/ui-core/lib';
3
3
  import { jsx, jsxs } from 'react/jsx-runtime';
4
4
  import { AlertCircle } from 'lucide-react';
@@ -55,5 +55,5 @@ function createDemoTree({
55
55
  __name(createDemoTree, "createDemoTree");
56
56
 
57
57
  export { TreeError, TreeSkeleton, createDemoTree };
58
- //# sourceMappingURL=chunk-KR6B3LVY.mjs.map
59
- //# sourceMappingURL=chunk-KR6B3LVY.mjs.map
58
+ //# sourceMappingURL=chunk-B6IR5KSC.mjs.map
59
+ //# sourceMappingURL=chunk-B6IR5KSC.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/tools/Tree/components/TreeSkeleton.tsx","../src/tools/Tree/components/TreeError.tsx","../src/tools/Tree/data/createDemoTree.ts"],"names":["jsxs","cn","jsx"],"mappings":"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","file":"chunk-KR6B3LVY.mjs","sourcesContent":["'use client';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nexport interface TreeSkeletonProps {\n rows?: number;\n className?: string;\n}\n\nexport function TreeSkeleton({ rows = 6, className }: TreeSkeletonProps) {\n return (\n <div className={cn('flex flex-col gap-1 p-2', className)} aria-hidden>\n {Array.from({ length: rows }).map((_, i) => (\n <div key={i} className=\"flex items-center gap-2\" style={{ paddingLeft: (i % 3) * 16 }}>\n <span className=\"size-4 shrink-0 animate-pulse rounded bg-muted\" />\n <span className=\"size-4 shrink-0 animate-pulse rounded bg-muted\" />\n <span className=\"h-3 flex-1 animate-pulse rounded bg-muted\" />\n </div>\n ))}\n </div>\n );\n}\n","'use client';\n\nimport { AlertCircle } from 'lucide-react';\nimport { cn } from '@djangocfg/ui-core/lib';\n\nexport interface TreeErrorProps {\n children: React.ReactNode;\n className?: string;\n}\n\nexport function TreeError({ children, className }: TreeErrorProps) {\n return (\n <div\n className={cn(\n 'flex items-start gap-2 rounded-md border border-destructive/30 bg-destructive/10 px-3 py-2 text-sm text-destructive',\n className,\n )}\n role=\"alert\"\n >\n <AlertCircle aria-hidden className=\"mt-0.5 size-4 shrink-0\" />\n <span>{children}</span>\n </div>\n );\n}\n","'use client';\n\nimport type { TreeNode } from '../types';\n\nexport interface DemoNode {\n name: string;\n}\n\n/**\n * Build a deterministic synthetic tree for stories and tests.\n *\n * @example\n * const data = createDemoTree({ depth: 4, breadth: 3 });\n * <TreeRoot data={data} getItemName={(n) => n.data.name} />\n */\nexport function createDemoTree({\n depth = 3,\n breadth = 4,\n rootPrefix = 'node',\n}: {\n depth?: number;\n breadth?: number;\n rootPrefix?: string;\n} = {}): TreeNode<DemoNode>[] {\n const make = (id: string, name: string, level: number): TreeNode<DemoNode> => {\n if (level >= depth) {\n return { id, data: { name } };\n }\n const children: TreeNode<DemoNode>[] = [];\n for (let i = 0; i < breadth; i++) {\n const childId = `${id}/${i}`;\n children.push(make(childId, `${name}-${i}`, level + 1));\n }\n return { id, data: { name }, children };\n };\n\n const roots: TreeNode<DemoNode>[] = [];\n for (let i = 0; i < breadth; i++) {\n roots.push(make(`${rootPrefix}-${i}`, `${rootPrefix} ${i}`, 1));\n }\n return roots;\n}\n"]}
1
+ {"version":3,"sources":["../src/tools/Tree/components/TreeSkeleton.tsx","../src/tools/Tree/components/TreeError.tsx","../src/tools/Tree/data/createDemoTree.ts"],"names":["jsxs","cn","jsx"],"mappings":"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","file":"chunk-B6IR5KSC.mjs","sourcesContent":["'use client';\n\nimport { cn } from '@djangocfg/ui-core/lib';\n\nexport interface TreeSkeletonProps {\n rows?: number;\n className?: string;\n}\n\nexport function TreeSkeleton({ rows = 6, className }: TreeSkeletonProps) {\n return (\n <div className={cn('flex flex-col gap-1 p-2', className)} aria-hidden>\n {Array.from({ length: rows }).map((_, i) => (\n <div key={i} className=\"flex items-center gap-2\" style={{ paddingLeft: (i % 3) * 16 }}>\n <span className=\"size-4 shrink-0 animate-pulse rounded bg-muted\" />\n <span className=\"size-4 shrink-0 animate-pulse rounded bg-muted\" />\n <span className=\"h-3 flex-1 animate-pulse rounded bg-muted\" />\n </div>\n ))}\n </div>\n );\n}\n","'use client';\n\nimport { AlertCircle } from 'lucide-react';\nimport { cn } from '@djangocfg/ui-core/lib';\n\nexport interface TreeErrorProps {\n children: React.ReactNode;\n className?: string;\n}\n\nexport function TreeError({ children, className }: TreeErrorProps) {\n return (\n <div\n className={cn(\n 'flex items-start gap-2 rounded-md border border-destructive/30 bg-destructive/10 px-3 py-2 text-sm text-destructive',\n className,\n )}\n role=\"alert\"\n >\n <AlertCircle aria-hidden className=\"mt-0.5 size-4 shrink-0\" />\n <span>{children}</span>\n </div>\n );\n}\n","'use client';\n\nimport type { TreeNode } from '../types';\n\nexport interface DemoNode {\n name: string;\n}\n\n/**\n * Build a deterministic synthetic tree for stories and tests.\n *\n * @example\n * const data = createDemoTree({ depth: 4, breadth: 3 });\n * <TreeRoot data={data} getItemName={(n) => n.data.name} />\n */\nexport function createDemoTree({\n depth = 3,\n breadth = 4,\n rootPrefix = 'node',\n}: {\n depth?: number;\n breadth?: number;\n rootPrefix?: string;\n} = {}): TreeNode<DemoNode>[] {\n const make = (id: string, name: string, level: number): TreeNode<DemoNode> => {\n if (level >= depth) {\n return { id, data: { name } };\n }\n const children: TreeNode<DemoNode>[] = [];\n for (let i = 0; i < breadth; i++) {\n const childId = `${id}/${i}`;\n children.push(make(childId, `${name}-${i}`, level + 1));\n }\n return { id, data: { name }, children };\n };\n\n const roots: TreeNode<DemoNode>[] = [];\n for (let i = 0; i < breadth; i++) {\n roots.push(make(`${rootPrefix}-${i}`, `${rootPrefix} ${i}`, 1));\n }\n return roots;\n}\n"]}
@@ -1,4 +1,4 @@
1
- import { __name, __require } from './chunk-CGILA3WO.mjs';
1
+ import { __name, __require } from './chunk-N2XQF2OL.mjs';
2
2
  import { create } from 'zustand';
3
3
  import { devtools, persist } from 'zustand/middleware';
4
4
  import { useShallow } from 'zustand/react/shallow';
@@ -334,5 +334,5 @@ function generateContentKey(content) {
334
334
  __name(generateContentKey, "generateContentKey");
335
335
 
336
336
  export { generateContentKey, useAudioCache, useBlobUrlCleanup, useImageCache, useMediaCacheStore, useVideoCache, useVideoPlayerSettings };
337
- //# sourceMappingURL=chunk-5LBDYFWH.mjs.map
338
- //# sourceMappingURL=chunk-5LBDYFWH.mjs.map
337
+ //# sourceMappingURL=chunk-C6GXVH5J.mjs.map
338
+ //# sourceMappingURL=chunk-C6GXVH5J.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/stores/mediaCache.ts"],"names":[],"mappings":"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","file":"chunk-5LBDYFWH.mjs","sourcesContent":["'use client';\n\nimport { create } from 'zustand';\nimport { devtools, persist } from 'zustand/middleware';\nimport { useShallow } from 'zustand/react/shallow';\n\nimport { createLogger } from '@djangocfg/ui-core/lib';\n\nconst cacheDebug = createLogger('MediaCache');\n\n// Types\ninterface BlobUrlEntry {\n url: string;\n refCount: number;\n createdAt: number;\n}\n\ninterface ImageDimensions {\n width: number;\n height: number;\n}\n\ninterface VideoMetadata {\n duration: number;\n width: number;\n height: number;\n codec?: string;\n}\n\ninterface EffectConfig {\n opacity: number;\n scale: number;\n blur: string;\n}\n\n/** Video player settings (persisted) */\ninterface VideoPlayerSettings {\n volume: number;\n isLooping: boolean;\n}\n\n// Stream URL TTL (30 seconds - shorter to avoid stale session/token issues)\nconst STREAM_URL_TTL = 30 * 1000;\n\ninterface MediaCacheState {\n // Blob URL cache (shared by Image, Audio, Video)\n blobUrls: Map<string, BlobUrlEntry>;\n\n // Image-specific\n imageDimensions: Map<string, ImageDimensions>;\n\n // Audio-specific\n audioPlaybackPositions: Map<string, number>;\n audioEffectConfigs: Map<string, EffectConfig>;\n\n // Video-specific\n videoStreamUrls: Map<string, { url: string; timestamp: number }>;\n videoPosterUrls: Map<string, string>;\n videoPlaybackPositions: Map<string, number>;\n videoMetadata: Map<string, VideoMetadata>;\n videoPlayerSettings: VideoPlayerSettings;\n}\n\ninterface MediaCacheActions {\n // Blob URL management (shared)\n getOrCreateBlobUrl: (\n key: string,\n content: ArrayBuffer,\n mimeType: string\n ) => string;\n releaseBlobUrl: (key: string) => void;\n hasBlobUrl: (key: string) => boolean;\n\n // Image actions\n cacheDimensions: (src: string, dims: ImageDimensions) => void;\n getDimensions: (src: string) => ImageDimensions | null;\n\n // Audio actions\n saveAudioPosition: (uri: string, time: number) => void;\n getAudioPosition: (uri: string) => number | null;\n getEffectConfig: (key: string) => EffectConfig | null;\n cacheEffectConfig: (key: string, config: EffectConfig) => void;\n\n // Video actions\n getOrCreateStreamUrl: (\n sessionId: string,\n path: string,\n generator: (sessionId: string, path: string) => string\n ) => string;\n getPosterUrl: (title: string) => string | null;\n cachePosterUrl: (title: string, url: string) => void;\n saveVideoPosition: (key: string, time: number) => void;\n getVideoPosition: (key: string) => number | null;\n cacheVideoMetadata: (url: string, meta: VideoMetadata) => void;\n getVideoMetadata: (url: string) => VideoMetadata | null;\n invalidateSession: (sessionId: string) => void;\n getVideoPlayerSettings: () => VideoPlayerSettings;\n saveVideoPlayerSettings: (settings: Partial<VideoPlayerSettings>) => void;\n\n // Global\n clearCache: () => void;\n getCacheStats: () => {\n blobUrls: number;\n dimensions: number;\n audioPositions: number;\n videoPositions: number;\n };\n}\n\ntype MediaCacheStore = MediaCacheState & MediaCacheActions;\n\n// Initial state\nconst DEFAULT_VIDEO_SETTINGS: VideoPlayerSettings = {\n volume: 1,\n isLooping: false,\n};\n\nconst initialState: MediaCacheState = {\n blobUrls: new Map(),\n imageDimensions: new Map(),\n audioPlaybackPositions: new Map(),\n audioEffectConfigs: new Map(),\n videoStreamUrls: new Map(),\n videoPosterUrls: new Map(),\n videoPlaybackPositions: new Map(),\n videoMetadata: new Map(),\n videoPlayerSettings: DEFAULT_VIDEO_SETTINGS,\n};\n\nexport const useMediaCacheStore = create<MediaCacheStore>()(\n devtools(\n persist(\n (set, get) => ({\n ...initialState,\n\n // ========== Blob URL Management ==========\n\n getOrCreateBlobUrl: (key, content, mimeType) => {\n const existing = get().blobUrls.get(key);\n if (existing) {\n // Increment ref count\n cacheDebug.debug(`Blob URL reused: ${key}`, { refCount: existing.refCount + 1 });\n set(\n (state) => ({\n blobUrls: new Map(state.blobUrls).set(key, {\n ...existing,\n refCount: existing.refCount + 1,\n }),\n }),\n false,\n 'blobUrl/incrementRef'\n );\n return existing.url;\n }\n\n // Create new blob URL\n const blob = new Blob([content], { type: mimeType });\n const url = URL.createObjectURL(blob);\n const sizeMB = (content.byteLength / 1024 / 1024).toFixed(2);\n cacheDebug.debug(`Blob URL created: ${key}`, { mimeType, size: `${sizeMB}MB` });\n set(\n (state) => ({\n blobUrls: new Map(state.blobUrls).set(key, {\n url,\n refCount: 1,\n createdAt: Date.now(),\n }),\n }),\n false,\n 'blobUrl/create'\n );\n return url;\n },\n\n releaseBlobUrl: (key) => {\n const entry = get().blobUrls.get(key);\n if (!entry) return;\n\n if (entry.refCount <= 1) {\n // Last reference - revoke and remove\n cacheDebug.debug(`Blob URL revoked: ${key}`);\n URL.revokeObjectURL(entry.url);\n set(\n (state) => {\n const newMap = new Map(state.blobUrls);\n newMap.delete(key);\n return { blobUrls: newMap };\n },\n false,\n 'blobUrl/revoke'\n );\n } else {\n // Decrement ref count\n cacheDebug.debug(`Blob URL released: ${key}`, { refCount: entry.refCount - 1 });\n set(\n (state) => ({\n blobUrls: new Map(state.blobUrls).set(key, {\n ...entry,\n refCount: entry.refCount - 1,\n }),\n }),\n false,\n 'blobUrl/decrementRef'\n );\n }\n },\n\n hasBlobUrl: (key) => get().blobUrls.has(key),\n\n // ========== Image Cache ==========\n\n cacheDimensions: (src, dims) => {\n set(\n (state) => ({\n imageDimensions: new Map(state.imageDimensions).set(src, dims),\n }),\n false,\n 'image/cacheDimensions'\n );\n },\n\n getDimensions: (src) => get().imageDimensions.get(src) ?? null,\n\n // ========== Audio Cache ==========\n\n saveAudioPosition: (uri, time) => {\n set(\n (state) => ({\n audioPlaybackPositions: new Map(state.audioPlaybackPositions).set(\n uri,\n time\n ),\n }),\n false,\n 'audio/savePosition'\n );\n },\n\n getAudioPosition: (uri) =>\n get().audioPlaybackPositions.get(uri) ?? null,\n\n getEffectConfig: (key) => get().audioEffectConfigs.get(key) ?? null,\n\n cacheEffectConfig: (key, config) => {\n set(\n (state) => ({\n audioEffectConfigs: new Map(state.audioEffectConfigs).set(\n key,\n config\n ),\n }),\n false,\n 'audio/cacheEffectConfig'\n );\n },\n\n // ========== Video Cache ==========\n\n getOrCreateStreamUrl: (sessionId, path, generator) => {\n const key = `${sessionId}:${path}`;\n const cached = get().videoStreamUrls.get(key);\n\n // Return if fresh\n if (cached && Date.now() - cached.timestamp < STREAM_URL_TTL) {\n return cached.url;\n }\n\n // Generate and cache\n const url = generator(sessionId, path);\n set(\n (state) => ({\n videoStreamUrls: new Map(state.videoStreamUrls).set(key, {\n url,\n timestamp: Date.now(),\n }),\n }),\n false,\n 'video/cacheStreamUrl'\n );\n return url;\n },\n\n getPosterUrl: (title) => get().videoPosterUrls.get(title) ?? null,\n\n cachePosterUrl: (title, url) => {\n set(\n (state) => ({\n videoPosterUrls: new Map(state.videoPosterUrls).set(title, url),\n }),\n false,\n 'video/cachePosterUrl'\n );\n },\n\n saveVideoPosition: (key, time) => {\n set(\n (state) => ({\n videoPlaybackPositions: new Map(state.videoPlaybackPositions).set(\n key,\n time\n ),\n }),\n false,\n 'video/savePosition'\n );\n },\n\n getVideoPosition: (key) =>\n get().videoPlaybackPositions.get(key) ?? null,\n\n cacheVideoMetadata: (url, meta) => {\n set(\n (state) => ({\n videoMetadata: new Map(state.videoMetadata).set(url, meta),\n }),\n false,\n 'video/cacheMetadata'\n );\n },\n\n getVideoMetadata: (url) => get().videoMetadata.get(url) ?? null,\n\n invalidateSession: (sessionId) => {\n set(\n (state) => {\n const newUrls = new Map(state.videoStreamUrls);\n for (const [key] of newUrls) {\n if (key.startsWith(`${sessionId}:`)) {\n newUrls.delete(key);\n }\n }\n return { videoStreamUrls: newUrls };\n },\n false,\n 'video/invalidateSession'\n );\n },\n\n getVideoPlayerSettings: () => get().videoPlayerSettings,\n\n saveVideoPlayerSettings: (settings) => {\n set(\n (state) => ({\n videoPlayerSettings: { ...state.videoPlayerSettings, ...settings },\n }),\n false,\n 'video/savePlayerSettings'\n );\n },\n\n // ========== Global ==========\n\n clearCache: () => {\n const stats = get().getCacheStats();\n cacheDebug.info('Clearing cache', stats);\n // Revoke all blob URLs before clearing\n get().blobUrls.forEach(({ url }) => URL.revokeObjectURL(url));\n set(initialState, false, 'clearCache');\n },\n\n getCacheStats: () => ({\n blobUrls: get().blobUrls.size,\n dimensions: get().imageDimensions.size,\n audioPositions: get().audioPlaybackPositions.size,\n videoPositions: get().videoPlaybackPositions.size,\n }),\n }),\n {\n name: 'media-cache-storage',\n // Only persist playback positions, poster URLs, and player settings\n partialize: (state) => ({\n audioPlaybackPositions: Array.from(\n state.audioPlaybackPositions.entries()\n ),\n videoPlaybackPositions: Array.from(\n state.videoPlaybackPositions.entries()\n ),\n videoPosterUrls: Array.from(state.videoPosterUrls.entries()),\n videoPlayerSettings: state.videoPlayerSettings,\n }),\n // Rehydrate Maps from arrays\n onRehydrateStorage: () => (state) => {\n if (state) {\n // Type assertion for persisted data\n const persistedAudio = state.audioPlaybackPositions as unknown;\n const persistedVideo = state.videoPlaybackPositions as unknown;\n const persistedPosters = state.videoPosterUrls as unknown;\n const persistedSettings = state.videoPlayerSettings as unknown;\n\n state.audioPlaybackPositions = new Map(\n Array.isArray(persistedAudio)\n ? (persistedAudio as [string, number][])\n : []\n );\n state.videoPlaybackPositions = new Map(\n Array.isArray(persistedVideo)\n ? (persistedVideo as [string, number][])\n : []\n );\n state.videoPosterUrls = new Map(\n Array.isArray(persistedPosters)\n ? (persistedPosters as [string, string][])\n : []\n );\n // Merge persisted settings with defaults (handles missing fields)\n state.videoPlayerSettings = {\n ...DEFAULT_VIDEO_SETTINGS,\n ...(persistedSettings && typeof persistedSettings === 'object'\n ? (persistedSettings as Partial<VideoPlayerSettings>)\n : {}),\n };\n }\n },\n }\n ),\n { name: 'MediaCacheStore' }\n )\n);\n\n// ========== Selective Hooks ==========\n\n/**\n * Hook for image-related cache operations\n * Uses useShallow to prevent infinite re-renders\n */\nexport const useImageCache = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n getOrCreateBlobUrl: state.getOrCreateBlobUrl,\n releaseBlobUrl: state.releaseBlobUrl,\n hasBlobUrl: state.hasBlobUrl,\n cacheDimensions: state.cacheDimensions,\n getDimensions: state.getDimensions,\n }))\n );\n\n/**\n * Hook for audio-related cache operations\n * Uses useShallow to prevent infinite re-renders\n */\nexport const useAudioCache = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n getOrCreateBlobUrl: state.getOrCreateBlobUrl,\n releaseBlobUrl: state.releaseBlobUrl,\n hasBlobUrl: state.hasBlobUrl,\n saveAudioPosition: state.saveAudioPosition,\n getAudioPosition: state.getAudioPosition,\n getEffectConfig: state.getEffectConfig,\n cacheEffectConfig: state.cacheEffectConfig,\n }))\n );\n\n/**\n * Hook for video-related cache operations\n * Uses useShallow to prevent infinite re-renders\n */\nexport const useVideoCache = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n getOrCreateBlobUrl: state.getOrCreateBlobUrl,\n releaseBlobUrl: state.releaseBlobUrl,\n hasBlobUrl: state.hasBlobUrl,\n getOrCreateStreamUrl: state.getOrCreateStreamUrl,\n getPosterUrl: state.getPosterUrl,\n cachePosterUrl: state.cachePosterUrl,\n saveVideoPosition: state.saveVideoPosition,\n getVideoPosition: state.getVideoPosition,\n cacheVideoMetadata: state.cacheVideoMetadata,\n getVideoMetadata: state.getVideoMetadata,\n invalidateSession: state.invalidateSession,\n getVideoPlayerSettings: state.getVideoPlayerSettings,\n saveVideoPlayerSettings: state.saveVideoPlayerSettings,\n }))\n );\n\n/**\n * Hook for video player settings only\n * Returns current settings and save function\n */\nexport const useVideoPlayerSettings = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n settings: state.videoPlayerSettings,\n saveSettings: state.saveVideoPlayerSettings,\n }))\n );\n\n/**\n * Hook for blob URL cleanup on unmount\n */\nexport function useBlobUrlCleanup(key: string | null) {\n const releaseBlobUrl = useMediaCacheStore((s) => s.releaseBlobUrl);\n\n // Using inline effect to avoid importing useEffect in store\n if (typeof window !== 'undefined') {\n // eslint-disable-next-line react-hooks/rules-of-hooks\n const { useEffect } = require('react');\n // eslint-disable-next-line react-hooks/rules-of-hooks\n useEffect(() => {\n return () => {\n if (key) {\n releaseBlobUrl(key);\n }\n };\n }, [key, releaseBlobUrl]);\n }\n}\n\n// ========== Utilities ==========\n\n/**\n * Generate a cache key from ArrayBuffer content\n * Uses first and last 1KB + length for fast hashing\n */\nexport function generateContentKey(content: ArrayBuffer): string {\n const arr = new Uint8Array(content);\n const len = arr.length;\n\n // Take first 1KB\n const start = arr.slice(0, Math.min(1024, len));\n // Take last 1KB\n const end = arr.slice(Math.max(0, len - 1024));\n\n // Simple hash from bytes\n let hash = len;\n for (let i = 0; i < start.length; i++) {\n hash = ((hash << 5) - hash + start[i]) | 0;\n }\n for (let i = 0; i < end.length; i++) {\n hash = ((hash << 5) - hash + end[i]) | 0;\n }\n\n return `blob-${len}-${hash.toString(16)}`;\n}\n"]}
1
+ {"version":3,"sources":["../src/stores/mediaCache.ts"],"names":[],"mappings":"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","file":"chunk-C6GXVH5J.mjs","sourcesContent":["'use client';\n\nimport { create } from 'zustand';\nimport { devtools, persist } from 'zustand/middleware';\nimport { useShallow } from 'zustand/react/shallow';\n\nimport { createLogger } from '@djangocfg/ui-core/lib';\n\nconst cacheDebug = createLogger('MediaCache');\n\n// Types\ninterface BlobUrlEntry {\n url: string;\n refCount: number;\n createdAt: number;\n}\n\ninterface ImageDimensions {\n width: number;\n height: number;\n}\n\ninterface VideoMetadata {\n duration: number;\n width: number;\n height: number;\n codec?: string;\n}\n\ninterface EffectConfig {\n opacity: number;\n scale: number;\n blur: string;\n}\n\n/** Video player settings (persisted) */\ninterface VideoPlayerSettings {\n volume: number;\n isLooping: boolean;\n}\n\n// Stream URL TTL (30 seconds - shorter to avoid stale session/token issues)\nconst STREAM_URL_TTL = 30 * 1000;\n\ninterface MediaCacheState {\n // Blob URL cache (shared by Image, Audio, Video)\n blobUrls: Map<string, BlobUrlEntry>;\n\n // Image-specific\n imageDimensions: Map<string, ImageDimensions>;\n\n // Audio-specific\n audioPlaybackPositions: Map<string, number>;\n audioEffectConfigs: Map<string, EffectConfig>;\n\n // Video-specific\n videoStreamUrls: Map<string, { url: string; timestamp: number }>;\n videoPosterUrls: Map<string, string>;\n videoPlaybackPositions: Map<string, number>;\n videoMetadata: Map<string, VideoMetadata>;\n videoPlayerSettings: VideoPlayerSettings;\n}\n\ninterface MediaCacheActions {\n // Blob URL management (shared)\n getOrCreateBlobUrl: (\n key: string,\n content: ArrayBuffer,\n mimeType: string\n ) => string;\n releaseBlobUrl: (key: string) => void;\n hasBlobUrl: (key: string) => boolean;\n\n // Image actions\n cacheDimensions: (src: string, dims: ImageDimensions) => void;\n getDimensions: (src: string) => ImageDimensions | null;\n\n // Audio actions\n saveAudioPosition: (uri: string, time: number) => void;\n getAudioPosition: (uri: string) => number | null;\n getEffectConfig: (key: string) => EffectConfig | null;\n cacheEffectConfig: (key: string, config: EffectConfig) => void;\n\n // Video actions\n getOrCreateStreamUrl: (\n sessionId: string,\n path: string,\n generator: (sessionId: string, path: string) => string\n ) => string;\n getPosterUrl: (title: string) => string | null;\n cachePosterUrl: (title: string, url: string) => void;\n saveVideoPosition: (key: string, time: number) => void;\n getVideoPosition: (key: string) => number | null;\n cacheVideoMetadata: (url: string, meta: VideoMetadata) => void;\n getVideoMetadata: (url: string) => VideoMetadata | null;\n invalidateSession: (sessionId: string) => void;\n getVideoPlayerSettings: () => VideoPlayerSettings;\n saveVideoPlayerSettings: (settings: Partial<VideoPlayerSettings>) => void;\n\n // Global\n clearCache: () => void;\n getCacheStats: () => {\n blobUrls: number;\n dimensions: number;\n audioPositions: number;\n videoPositions: number;\n };\n}\n\ntype MediaCacheStore = MediaCacheState & MediaCacheActions;\n\n// Initial state\nconst DEFAULT_VIDEO_SETTINGS: VideoPlayerSettings = {\n volume: 1,\n isLooping: false,\n};\n\nconst initialState: MediaCacheState = {\n blobUrls: new Map(),\n imageDimensions: new Map(),\n audioPlaybackPositions: new Map(),\n audioEffectConfigs: new Map(),\n videoStreamUrls: new Map(),\n videoPosterUrls: new Map(),\n videoPlaybackPositions: new Map(),\n videoMetadata: new Map(),\n videoPlayerSettings: DEFAULT_VIDEO_SETTINGS,\n};\n\nexport const useMediaCacheStore = create<MediaCacheStore>()(\n devtools(\n persist(\n (set, get) => ({\n ...initialState,\n\n // ========== Blob URL Management ==========\n\n getOrCreateBlobUrl: (key, content, mimeType) => {\n const existing = get().blobUrls.get(key);\n if (existing) {\n // Increment ref count\n cacheDebug.debug(`Blob URL reused: ${key}`, { refCount: existing.refCount + 1 });\n set(\n (state) => ({\n blobUrls: new Map(state.blobUrls).set(key, {\n ...existing,\n refCount: existing.refCount + 1,\n }),\n }),\n false,\n 'blobUrl/incrementRef'\n );\n return existing.url;\n }\n\n // Create new blob URL\n const blob = new Blob([content], { type: mimeType });\n const url = URL.createObjectURL(blob);\n const sizeMB = (content.byteLength / 1024 / 1024).toFixed(2);\n cacheDebug.debug(`Blob URL created: ${key}`, { mimeType, size: `${sizeMB}MB` });\n set(\n (state) => ({\n blobUrls: new Map(state.blobUrls).set(key, {\n url,\n refCount: 1,\n createdAt: Date.now(),\n }),\n }),\n false,\n 'blobUrl/create'\n );\n return url;\n },\n\n releaseBlobUrl: (key) => {\n const entry = get().blobUrls.get(key);\n if (!entry) return;\n\n if (entry.refCount <= 1) {\n // Last reference - revoke and remove\n cacheDebug.debug(`Blob URL revoked: ${key}`);\n URL.revokeObjectURL(entry.url);\n set(\n (state) => {\n const newMap = new Map(state.blobUrls);\n newMap.delete(key);\n return { blobUrls: newMap };\n },\n false,\n 'blobUrl/revoke'\n );\n } else {\n // Decrement ref count\n cacheDebug.debug(`Blob URL released: ${key}`, { refCount: entry.refCount - 1 });\n set(\n (state) => ({\n blobUrls: new Map(state.blobUrls).set(key, {\n ...entry,\n refCount: entry.refCount - 1,\n }),\n }),\n false,\n 'blobUrl/decrementRef'\n );\n }\n },\n\n hasBlobUrl: (key) => get().blobUrls.has(key),\n\n // ========== Image Cache ==========\n\n cacheDimensions: (src, dims) => {\n set(\n (state) => ({\n imageDimensions: new Map(state.imageDimensions).set(src, dims),\n }),\n false,\n 'image/cacheDimensions'\n );\n },\n\n getDimensions: (src) => get().imageDimensions.get(src) ?? null,\n\n // ========== Audio Cache ==========\n\n saveAudioPosition: (uri, time) => {\n set(\n (state) => ({\n audioPlaybackPositions: new Map(state.audioPlaybackPositions).set(\n uri,\n time\n ),\n }),\n false,\n 'audio/savePosition'\n );\n },\n\n getAudioPosition: (uri) =>\n get().audioPlaybackPositions.get(uri) ?? null,\n\n getEffectConfig: (key) => get().audioEffectConfigs.get(key) ?? null,\n\n cacheEffectConfig: (key, config) => {\n set(\n (state) => ({\n audioEffectConfigs: new Map(state.audioEffectConfigs).set(\n key,\n config\n ),\n }),\n false,\n 'audio/cacheEffectConfig'\n );\n },\n\n // ========== Video Cache ==========\n\n getOrCreateStreamUrl: (sessionId, path, generator) => {\n const key = `${sessionId}:${path}`;\n const cached = get().videoStreamUrls.get(key);\n\n // Return if fresh\n if (cached && Date.now() - cached.timestamp < STREAM_URL_TTL) {\n return cached.url;\n }\n\n // Generate and cache\n const url = generator(sessionId, path);\n set(\n (state) => ({\n videoStreamUrls: new Map(state.videoStreamUrls).set(key, {\n url,\n timestamp: Date.now(),\n }),\n }),\n false,\n 'video/cacheStreamUrl'\n );\n return url;\n },\n\n getPosterUrl: (title) => get().videoPosterUrls.get(title) ?? null,\n\n cachePosterUrl: (title, url) => {\n set(\n (state) => ({\n videoPosterUrls: new Map(state.videoPosterUrls).set(title, url),\n }),\n false,\n 'video/cachePosterUrl'\n );\n },\n\n saveVideoPosition: (key, time) => {\n set(\n (state) => ({\n videoPlaybackPositions: new Map(state.videoPlaybackPositions).set(\n key,\n time\n ),\n }),\n false,\n 'video/savePosition'\n );\n },\n\n getVideoPosition: (key) =>\n get().videoPlaybackPositions.get(key) ?? null,\n\n cacheVideoMetadata: (url, meta) => {\n set(\n (state) => ({\n videoMetadata: new Map(state.videoMetadata).set(url, meta),\n }),\n false,\n 'video/cacheMetadata'\n );\n },\n\n getVideoMetadata: (url) => get().videoMetadata.get(url) ?? null,\n\n invalidateSession: (sessionId) => {\n set(\n (state) => {\n const newUrls = new Map(state.videoStreamUrls);\n for (const [key] of newUrls) {\n if (key.startsWith(`${sessionId}:`)) {\n newUrls.delete(key);\n }\n }\n return { videoStreamUrls: newUrls };\n },\n false,\n 'video/invalidateSession'\n );\n },\n\n getVideoPlayerSettings: () => get().videoPlayerSettings,\n\n saveVideoPlayerSettings: (settings) => {\n set(\n (state) => ({\n videoPlayerSettings: { ...state.videoPlayerSettings, ...settings },\n }),\n false,\n 'video/savePlayerSettings'\n );\n },\n\n // ========== Global ==========\n\n clearCache: () => {\n const stats = get().getCacheStats();\n cacheDebug.info('Clearing cache', stats);\n // Revoke all blob URLs before clearing\n get().blobUrls.forEach(({ url }) => URL.revokeObjectURL(url));\n set(initialState, false, 'clearCache');\n },\n\n getCacheStats: () => ({\n blobUrls: get().blobUrls.size,\n dimensions: get().imageDimensions.size,\n audioPositions: get().audioPlaybackPositions.size,\n videoPositions: get().videoPlaybackPositions.size,\n }),\n }),\n {\n name: 'media-cache-storage',\n // Only persist playback positions, poster URLs, and player settings\n partialize: (state) => ({\n audioPlaybackPositions: Array.from(\n state.audioPlaybackPositions.entries()\n ),\n videoPlaybackPositions: Array.from(\n state.videoPlaybackPositions.entries()\n ),\n videoPosterUrls: Array.from(state.videoPosterUrls.entries()),\n videoPlayerSettings: state.videoPlayerSettings,\n }),\n // Rehydrate Maps from arrays\n onRehydrateStorage: () => (state) => {\n if (state) {\n // Type assertion for persisted data\n const persistedAudio = state.audioPlaybackPositions as unknown;\n const persistedVideo = state.videoPlaybackPositions as unknown;\n const persistedPosters = state.videoPosterUrls as unknown;\n const persistedSettings = state.videoPlayerSettings as unknown;\n\n state.audioPlaybackPositions = new Map(\n Array.isArray(persistedAudio)\n ? (persistedAudio as [string, number][])\n : []\n );\n state.videoPlaybackPositions = new Map(\n Array.isArray(persistedVideo)\n ? (persistedVideo as [string, number][])\n : []\n );\n state.videoPosterUrls = new Map(\n Array.isArray(persistedPosters)\n ? (persistedPosters as [string, string][])\n : []\n );\n // Merge persisted settings with defaults (handles missing fields)\n state.videoPlayerSettings = {\n ...DEFAULT_VIDEO_SETTINGS,\n ...(persistedSettings && typeof persistedSettings === 'object'\n ? (persistedSettings as Partial<VideoPlayerSettings>)\n : {}),\n };\n }\n },\n }\n ),\n { name: 'MediaCacheStore' }\n )\n);\n\n// ========== Selective Hooks ==========\n\n/**\n * Hook for image-related cache operations\n * Uses useShallow to prevent infinite re-renders\n */\nexport const useImageCache = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n getOrCreateBlobUrl: state.getOrCreateBlobUrl,\n releaseBlobUrl: state.releaseBlobUrl,\n hasBlobUrl: state.hasBlobUrl,\n cacheDimensions: state.cacheDimensions,\n getDimensions: state.getDimensions,\n }))\n );\n\n/**\n * Hook for audio-related cache operations\n * Uses useShallow to prevent infinite re-renders\n */\nexport const useAudioCache = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n getOrCreateBlobUrl: state.getOrCreateBlobUrl,\n releaseBlobUrl: state.releaseBlobUrl,\n hasBlobUrl: state.hasBlobUrl,\n saveAudioPosition: state.saveAudioPosition,\n getAudioPosition: state.getAudioPosition,\n getEffectConfig: state.getEffectConfig,\n cacheEffectConfig: state.cacheEffectConfig,\n }))\n );\n\n/**\n * Hook for video-related cache operations\n * Uses useShallow to prevent infinite re-renders\n */\nexport const useVideoCache = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n getOrCreateBlobUrl: state.getOrCreateBlobUrl,\n releaseBlobUrl: state.releaseBlobUrl,\n hasBlobUrl: state.hasBlobUrl,\n getOrCreateStreamUrl: state.getOrCreateStreamUrl,\n getPosterUrl: state.getPosterUrl,\n cachePosterUrl: state.cachePosterUrl,\n saveVideoPosition: state.saveVideoPosition,\n getVideoPosition: state.getVideoPosition,\n cacheVideoMetadata: state.cacheVideoMetadata,\n getVideoMetadata: state.getVideoMetadata,\n invalidateSession: state.invalidateSession,\n getVideoPlayerSettings: state.getVideoPlayerSettings,\n saveVideoPlayerSettings: state.saveVideoPlayerSettings,\n }))\n );\n\n/**\n * Hook for video player settings only\n * Returns current settings and save function\n */\nexport const useVideoPlayerSettings = () =>\n useMediaCacheStore(\n useShallow((state) => ({\n settings: state.videoPlayerSettings,\n saveSettings: state.saveVideoPlayerSettings,\n }))\n );\n\n/**\n * Hook for blob URL cleanup on unmount\n */\nexport function useBlobUrlCleanup(key: string | null) {\n const releaseBlobUrl = useMediaCacheStore((s) => s.releaseBlobUrl);\n\n // Using inline effect to avoid importing useEffect in store\n if (typeof window !== 'undefined') {\n // eslint-disable-next-line react-hooks/rules-of-hooks\n const { useEffect } = require('react');\n // eslint-disable-next-line react-hooks/rules-of-hooks\n useEffect(() => {\n return () => {\n if (key) {\n releaseBlobUrl(key);\n }\n };\n }, [key, releaseBlobUrl]);\n }\n}\n\n// ========== Utilities ==========\n\n/**\n * Generate a cache key from ArrayBuffer content\n * Uses first and last 1KB + length for fast hashing\n */\nexport function generateContentKey(content: ArrayBuffer): string {\n const arr = new Uint8Array(content);\n const len = arr.length;\n\n // Take first 1KB\n const start = arr.slice(0, Math.min(1024, len));\n // Take last 1KB\n const end = arr.slice(Math.max(0, len - 1024));\n\n // Simple hash from bytes\n let hash = len;\n for (let i = 0; i < start.length; i++) {\n hash = ((hash << 5) - hash + start[i]) | 0;\n }\n for (let i = 0; i < end.length; i++) {\n hash = ((hash << 5) - hash + end[i]) | 0;\n }\n\n return `blob-${len}-${hash.toString(16)}`;\n}\n"]}