@marimo-team/frontend 0.21.2-dev5 → 0.21.2-dev52

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 (386) hide show
  1. package/dist/assets/{CellStatus-CDSBsjjF.js → CellStatus-DX8MnBMk.js} +1 -1
  2. package/dist/assets/{ConnectedDataExplorerComponent-Buaffg3u.js → ConnectedDataExplorerComponent-7yXYooWG.js} +1 -1
  3. package/dist/assets/{DeferredRequestRegistry-O6RDJKs0.js → DeferredRequestRegistry-B8jPCuU1.js} +1 -1
  4. package/dist/assets/{ImperativeModal-qEtcJ95s.js → ImperativeModal-C3CD5-Aq.js} +1 -1
  5. package/dist/assets/JsonOutput-DqjAyzM8.js +46 -0
  6. package/dist/assets/{LazyAnyLanguageCodeMirror-m8w66E4s.js → LazyAnyLanguageCodeMirror-CcCHIX38.js} +2 -2
  7. package/dist/assets/{MarimoErrorOutput-az74f3Mp.js → MarimoErrorOutput-Dat_S09q.js} +5 -5
  8. package/dist/assets/RenderHTML-C7XM39M2.js +1 -0
  9. package/dist/assets/{add-cell-with-ai-DUn4LN4W.js → add-cell-with-ai-y2xroEGf.js} +9 -9
  10. package/dist/assets/{add-connection-dialog-CBz0AUVX.js → add-connection-dialog-QDBOmil3.js} +23 -23
  11. package/dist/assets/{agent-panel-u83dsDsM.js → agent-panel-DziFrEjJ.js} +6 -6
  12. package/dist/assets/ai-model-dropdown-CiJpv5wN.js +5 -0
  13. package/dist/assets/{alert-dialog-9WfvUF7e.js → alert-dialog-ebk_-wsU.js} +1 -1
  14. package/dist/assets/{any-language-editor-DwQMnAM3.js → any-language-editor-DZYnKlJB.js} +1 -1
  15. package/dist/assets/{app-config-button-rDu-lCIB.js → app-config-button-CpAv9Tiu.js} +1 -1
  16. package/dist/assets/{architectureDiagram-VXUJARFQ-DfMHHFZ3.js → architectureDiagram-VXUJARFQ-DKmk99ac.js} +1 -1
  17. package/dist/assets/{azure-BTXHztTw.js → azure-Dw5gKmUz.js} +1 -1
  18. package/dist/assets/{blockDiagram-VD42YOAC-CuEDwbg7.js → blockDiagram-VD42YOAC-TpAhP5OQ.js} +1 -1
  19. package/dist/assets/{c4Diagram-YG6GDRKO-CD9uz7WG.js → c4Diagram-YG6GDRKO-Dg8hCyKf.js} +1 -1
  20. package/dist/assets/{cache-panel-qIxnndwr.js → cache-panel-D0OsHk_D.js} +1 -1
  21. package/dist/assets/cell-editor-RbAVpSFb.js +22 -0
  22. package/dist/assets/{cell-link-Q_UEP94Z.js → cell-link-CNgO3c-T.js} +1 -1
  23. package/dist/assets/{cells-fq1RMGX_.js → cells-39RE4UzS.js} +77 -76
  24. package/dist/assets/channel-D1wPYhc9.js +1 -0
  25. package/dist/assets/{chat-display-D_Fuuy6r.js → chat-display-BPUVrlA6.js} +1 -1
  26. package/dist/assets/{chat-panel-BO-JoQSp.js → chat-panel-esz5TfF-.js} +2 -2
  27. package/dist/assets/{chunk-5FQGJX7Z-CM1a_K7I.js → chunk-5FQGJX7Z-sGWxL7Ey.js} +3 -3
  28. package/dist/assets/{chunk-ABZYJK2D-D_Wumzom.js → chunk-ABZYJK2D-DXJ7hnjx.js} +1 -1
  29. package/dist/assets/{chunk-ATLVNIR6-BZ0lTgIn.js → chunk-ATLVNIR6-CTEqcLEc.js} +1 -1
  30. package/dist/assets/{chunk-B4BG7PRW-Dw1S6osV.js → chunk-B4BG7PRW-CkAIqtPf.js} +1 -1
  31. package/dist/assets/{chunk-DI55MBZ5-Q63FTCPP.js → chunk-DI55MBZ5-C4F9vu8o.js} +1 -1
  32. package/dist/assets/{chunk-EXTU4WIE-CfkJ_R7W.js → chunk-EXTU4WIE-BT0Py-4P.js} +1 -1
  33. package/dist/assets/{chunk-JA3XYJ7Z-B41fGeoj.js → chunk-JA3XYJ7Z-CGd7jja_.js} +1 -1
  34. package/dist/assets/{chunk-JZLCHNYA-D43EJxQw.js → chunk-JZLCHNYA-CqY-RYTj.js} +1 -1
  35. package/dist/assets/{chunk-N4CR4FBY-DCtw0-ni.js → chunk-N4CR4FBY-aUDjVzN7.js} +2 -2
  36. package/dist/assets/{chunk-QN33PNHL-CLA2rsaS.js → chunk-QN33PNHL-DHnCb5-U.js} +1 -1
  37. package/dist/assets/{chunk-QXUST7PY-De1DpbnL.js → chunk-QXUST7PY-CRO5kYCE.js} +1 -1
  38. package/dist/assets/{chunk-S3R3BYOJ-qrwvlUdq.js → chunk-S3R3BYOJ-CSHQgcuP.js} +1 -1
  39. package/dist/assets/{chunk-TZMSLE5B-C4KcThb0.js → chunk-TZMSLE5B-CE7o9asH.js} +1 -1
  40. package/dist/assets/classDiagram-2ON5EDUG-BDIQSKYv.js +1 -0
  41. package/dist/assets/classDiagram-v2-WZHVMYZB-Cdh4cVOB.js +1 -0
  42. package/dist/assets/{code-block-37QAKDTI-D4FF4cnN.js → code-block-37QAKDTI-DMpe1jNG.js} +1 -1
  43. package/dist/assets/{column-preview-WsubyItX.js → column-preview-BUKHsglA.js} +1 -1
  44. package/dist/assets/{command-rfxSsiHU.js → command-DmzEbs6L.js} +1 -1
  45. package/dist/assets/{command-palette-Bez4BqJF.js → command-palette-BX5gNpUr.js} +1 -1
  46. package/dist/assets/{common-BYwO983A.js → common-B4yshY-H.js} +1 -1
  47. package/dist/assets/{components-Ozq0825a.js → components-DCqJqyys.js} +1 -1
  48. package/dist/assets/{components-Bpgzv_5_.js → components-oKCjx6cN.js} +1 -1
  49. package/dist/assets/{config-3Aq84phF.js → config-DqxNMQDN.js} +1 -1
  50. package/dist/assets/{context-CIAP2NOh.js → context-CRmPPhx9.js} +1 -1
  51. package/dist/assets/{context-aware-panel-CfsfBlhO.js → context-aware-panel-DC0Te9Hj.js} +2 -2
  52. package/dist/assets/copy-KjcPgPw9.js +1 -0
  53. package/dist/assets/{copy-icon-BgmMM9Zg.js → copy-icon-P0kzz1Pr.js} +1 -1
  54. package/dist/assets/{dagre-6UL2VRFP-BhMnPhuU.js → dagre-6UL2VRFP-XqoNLMQ3.js} +1 -1
  55. package/dist/assets/{data-grid-overlay-editor-kbv73SQ7.js → data-grid-overlay-editor-Cus1-BFN.js} +1 -1
  56. package/dist/assets/{datasource-F9sJtuof.js → datasource-DKHG39NV.js} +2 -2
  57. package/dist/assets/{dates-CHaNfieI.js → dates-CTqMeMGY.js} +1 -1
  58. package/dist/assets/{dependency-graph-panel-AOBcX2tP.js → dependency-graph-panel-7GKT3dyD.js} +4 -4
  59. package/dist/assets/{diagram-PSM6KHXK-Cr2t4zul.js → diagram-PSM6KHXK-BuzKNNcN.js} +1 -1
  60. package/dist/assets/{diagram-QEK2KX5R-DLU_hSO4.js → diagram-QEK2KX5R-A5LBQS11.js} +1 -1
  61. package/dist/assets/{diagram-S2PKOQOG-C7EkNAO3.js → diagram-S2PKOQOG-dkJE8SA0.js} +1 -1
  62. package/dist/assets/{dialog-CVN1lcMF.js → dialog-1_jIyAb_.js} +1 -1
  63. package/dist/assets/{dist-DJBpXGro.js → dist-6P6vjf93.js} +1 -1
  64. package/dist/assets/dist-BWGxTsTP.js +1 -0
  65. package/dist/assets/{dist-CblX3KGI.js → dist-B_buFm3B.js} +1 -1
  66. package/dist/assets/{dist-CXLJUPFl.js → dist-BatEf-0f.js} +1 -1
  67. package/dist/assets/dist-C3AnARPE.js +1 -0
  68. package/dist/assets/{dist-BNC_tnlW.js → dist-C6ohaMKU.js} +1 -1
  69. package/dist/assets/{dist-BoDHthoc.js → dist-CO9Pg2Rc.js} +1 -1
  70. package/dist/assets/{dist-By6zhkxZ.js → dist-CSV6H5w_.js} +1 -1
  71. package/dist/assets/{dist-6FIAeDIu.js → dist-CYs1oP7g.js} +1 -1
  72. package/dist/assets/{dist-DbK_4_qo.js → dist-DD9ZJMK5.js} +1 -1
  73. package/dist/assets/{dist-DlTbNRBn.js → dist-DReDpc2z.js} +1 -1
  74. package/dist/assets/dist-DSlq5ZeQ.js +1 -0
  75. package/dist/assets/{dist-DrceDAhM.js → dist-DXyIIB32.js} +1 -1
  76. package/dist/assets/{dist-CcGcwrUD.js → dist-DkaderFA.js} +1 -1
  77. package/dist/assets/{dist-BiZQlfhr.js → dist-Dq3BlK4R.js} +1 -1
  78. package/dist/assets/{dist-ClzRAVN-.js → dist-Dxov2ltb.js} +1 -1
  79. package/dist/assets/{dist-BXXPfIam.js → dist-EtGG9C1A.js} +1 -1
  80. package/dist/assets/{dist-C3MmXcLA.js → dist-NjId5Gdz.js} +1 -1
  81. package/dist/assets/dist-ZM59OWYL.js +1 -0
  82. package/dist/assets/dist-_TwZ4tIe.js +1 -0
  83. package/dist/assets/dist-f9b2H4q_.js +1 -0
  84. package/dist/assets/{dist-mxGLXOFm.js → dist-feWVQ4dq.js} +1 -1
  85. package/dist/assets/{documentation-panel-D28MpqbI.js → documentation-panel-D5d_BbHV.js} +1 -1
  86. package/dist/assets/{download-B9_ToVd4.js → download-B1vXaivj.js} +4 -4
  87. package/dist/assets/{dropdown-menu-DdPu5Kfu.js → dropdown-menu-BehqiLFL.js} +1 -1
  88. package/dist/assets/{edit-page-otS-QvnC.js → edit-page-B5cAZdO7.js} +7 -7
  89. package/dist/assets/{en-US-BO8E69bZ.js → en-US-f8tajDx1.js} +1 -1
  90. package/dist/assets/{erDiagram-Q2GNP2WA-DqlEo_Us.js → erDiagram-Q2GNP2WA-CgH6udOW.js} +1 -1
  91. package/dist/assets/{error-banner-BRG6IPhC.js → error-banner-J5F3weEj.js} +1 -1
  92. package/dist/assets/{error-panel-RRoj_iOR.js → error-panel-DnhKJ5Ou.js} +1 -1
  93. package/dist/assets/{es-CzkousWe.js → es-DYI7U61K.js} +1 -1
  94. package/dist/assets/{esm-CBkHtTOV.js → esm-BC1J92im.js} +1 -1
  95. package/dist/assets/{esm-DhTsVMLM.js → esm-DKLLCu3N.js} +1 -1
  96. package/dist/assets/{field-B1jGbQlH.js → field-B08lvxnl.js} +1 -1
  97. package/dist/assets/file-explorer-panel-BUk69TND.js +26 -0
  98. package/dist/assets/{file-icons-CWLHDa1J.js → file-icons-3DTS8ZM7.js} +1 -1
  99. package/dist/assets/{floating-outline-BvQKiC0F.js → floating-outline-DmugATcB.js} +1 -1
  100. package/dist/assets/{flowDiagram-NV44I4VS-CQlIJ12H.js → flowDiagram-NV44I4VS-D85BL4rA.js} +1 -1
  101. package/dist/assets/{focus-DYwTiH9-.js → focus-DlJH7sfh.js} +1 -1
  102. package/dist/assets/{form-CrQYrsUC.js → form-760_EqmF.js} +2 -2
  103. package/dist/assets/{formats-rhOJovGE.js → formats-wYuh1bqp.js} +1 -1
  104. package/dist/assets/{formatting-B4ZCH3ol.js → formatting-CraOPe94.js} +1 -1
  105. package/dist/assets/{gallery-page-eRea0yic.js → gallery-page--paiPvYX.js} +1 -1
  106. package/dist/assets/{ganttDiagram-JELNMOA3-B3EGIuZL.js → ganttDiagram-JELNMOA3-DzCOXWlF.js} +1 -1
  107. package/dist/assets/{gitGraphDiagram-V2S2FVAM-h5SlguJK.js → gitGraphDiagram-V2S2FVAM-BUHURX6t.js} +1 -1
  108. package/dist/assets/{glide-data-editor-xt5xNZeV.js → glide-data-editor-DYdoQTO5.js} +4 -4
  109. package/dist/assets/{globals-BsV5fVR-.js → globals-7M5DRsIb.js} +1 -1
  110. package/dist/assets/home-page-CxQ1e98h.js +4 -0
  111. package/dist/assets/{hooks-CHx5dUUq.js → hooks-C3jEff2O.js} +1 -1
  112. package/dist/assets/html-to-image-DTFXm6_Z.js +2 -0
  113. package/dist/assets/{index-GjXovVsl.css → index-BmoocKR0.css} +1 -1
  114. package/dist/assets/index-DNchElHl.js +38 -0
  115. package/dist/assets/{infoDiagram-HS3SLOUP-pzpClObZ.js → infoDiagram-HS3SLOUP-GG5r0Y5G.js} +1 -1
  116. package/dist/assets/{input-CUwqpKjd.js → input-CN1ZeRYm.js} +1 -1
  117. package/dist/assets/{isValid-BGe7pJXT.js → isValid-DY4Mgbr7.js} +1 -1
  118. package/dist/assets/{journeyDiagram-XKPGCS4Q-BrdRi92v.js → journeyDiagram-XKPGCS4Q-B1dvnV4D.js} +1 -1
  119. package/dist/assets/{kanban-definition-3W4ZIXB7-B6WqgFqh.js → kanban-definition-3W4ZIXB7-CefbrXQv.js} +1 -1
  120. package/dist/assets/{kiosk-mode-DNjEnIWb.js → kiosk-mode-CyTnrzwK.js} +1 -1
  121. package/dist/assets/{layout-CwFtCyW8.js → layout-3b3tzmbb.js} +3 -3
  122. package/dist/assets/{linear-BP9rwmWK.js → linear-BDnrHGYO.js} +1 -1
  123. package/dist/assets/{links-CIQwYQ48.js → links-CptYD1FP.js} +1 -1
  124. package/dist/assets/{logs-panel-BU1tNEVc.js → logs-panel-C4k_2IHW.js} +1 -1
  125. package/dist/assets/{markdown-renderer-MR9df58W.js → markdown-renderer-DSy5IuAx.js} +1 -1
  126. package/dist/assets/{menu-items-UxKrm8hS.js → menu-items-BpHXKv51.js} +1 -1
  127. package/dist/assets/mermaid-4DMBBIKO-g7AL4SLJ.js +1 -0
  128. package/dist/assets/{mermaid-Djr3jUAB.js → mermaid-C43VbdrM.js} +3 -3
  129. package/dist/assets/{mermaid-parser.core-DR82IMb2.js → mermaid-parser.core-CdM8D_p_.js} +1 -1
  130. package/dist/assets/{mindmap-definition-VGOIOE7T-DAmJpbPM.js → mindmap-definition-VGOIOE7T-CcV7D-wF.js} +1 -1
  131. package/dist/assets/{mode-AcL8c6qH.js → mode-D-iRbN9x.js} +1 -1
  132. package/dist/assets/{multi-map-Pp1P2DOX.js → multi-map-C8RbwBrw.js} +1 -1
  133. package/dist/assets/{name-cell-input-DVFEv_aO.js → name-cell-input-Dyb0ZO4X.js} +1 -1
  134. package/dist/assets/{number-overlay-editor-D__AflXQ.js → number-overlay-editor-BBJO1mf8.js} +1 -1
  135. package/dist/assets/{outline-panel-CP0oIuiU.js → outline-panel-D7CJwO8u.js} +1 -1
  136. package/dist/assets/{packages-panel-CzRELimG.js → packages-panel-Bp1SYjgc.js} +1 -1
  137. package/dist/assets/{panels-wTweRyIv.js → panels-D0aw6jdc.js} +1 -1
  138. package/dist/assets/{pieDiagram-ADFJNKIX-DbGIFRoq.js → pieDiagram-ADFJNKIX-C1fBf15W.js} +1 -1
  139. package/dist/assets/{popover-BPGG2gPG.js → popover-Bvoif-Mg.js} +1 -1
  140. package/dist/assets/{precisionRound-BOmLQIKI.js → precisionRound-CCOoIlcP.js} +1 -1
  141. package/dist/assets/{process-output-CTVsT--B.js → process-output-BxPWHcIi.js} +1 -1
  142. package/dist/assets/{quadrantDiagram-AYHSOK5B-CLAtyple.js → quadrantDiagram-AYHSOK5B-RVooN-fX.js} +1 -1
  143. package/dist/assets/{react-vega-BZav_-2n.js → react-vega-BKRQSy0g.js} +1 -1
  144. package/dist/assets/react-vega-DXESF7qN.js +1 -0
  145. package/dist/assets/{readonly-python-code-DH-1xZGq.js → readonly-python-code-D95oshJa.js} +1 -1
  146. package/dist/assets/{renderShortcut-CnD1Dah5.js → renderShortcut-BYvKm38e.js} +1 -1
  147. package/dist/assets/{request-registry-B-7cIM_I.js → request-registry-XB2EzJHm.js} +1 -1
  148. package/dist/assets/{requirementDiagram-UZGBJVZJ-DDDTB1LD.js → requirementDiagram-UZGBJVZJ-DtqOYJpH.js} +1 -1
  149. package/dist/assets/{run-page-FCvGnICa.js → run-page-B7CASued.js} +1 -1
  150. package/dist/assets/{sankeyDiagram-TZEHDZUN-DAWPOfBw.js → sankeyDiagram-TZEHDZUN-DmcooLA8.js} +1 -1
  151. package/dist/assets/{scratchpad-panel-k9lrm-rC.js → scratchpad-panel-BIVRgYOU.js} +1 -1
  152. package/dist/assets/{secrets-panel-B1Z-6dmz.js → secrets-panel-C24hZEiO.js} +1 -1
  153. package/dist/assets/{select-CxT2Geqr.js → select-Cnd3vm9n.js} +1 -1
  154. package/dist/assets/{sequenceDiagram-WL72ISMW-D_qAidD2.js → sequenceDiagram-WL72ISMW-S2qeLb7J.js} +1 -1
  155. package/dist/assets/{session-panel-WhN0qilM.js → session-panel-CNabBHQm.js} +1 -1
  156. package/dist/assets/{share-BdH_5I58.js → share-NjpZ54PJ.js} +1 -1
  157. package/dist/assets/{snippets-panel-BCDHUuku.js → snippets-panel-D1JAeJj1.js} +1 -1
  158. package/dist/assets/{spec-DbmSqx09.js → spec-DnVBmJUh.js} +1 -1
  159. package/dist/assets/{state-Ce7CVShi.js → state-BagvlaEz.js} +1 -1
  160. package/dist/assets/{state-3V5UxC3B.js → state-Cz_wrzCz.js} +1 -1
  161. package/dist/assets/{state-CBmCYWFH.js → state-DMQXxemS.js} +1 -1
  162. package/dist/assets/{stateDiagram-FKZM4ZOC-C7qgDrGA.js → stateDiagram-FKZM4ZOC-uuQ60pIl.js} +1 -1
  163. package/dist/assets/stateDiagram-v2-4FDKWEC3-D-GWhQTG.js +1 -0
  164. package/dist/assets/stex-CruVQx-P.js +1 -0
  165. package/dist/assets/{switch-CVKxYu_0.js → switch-Bwpd2AFq.js} +1 -1
  166. package/dist/assets/{terminal-Cr7wbEjz.js → terminal-Gl8Fi44y.js} +1 -1
  167. package/dist/assets/{time-D2GKc0U6.js → time-Bw8f15NM.js} +1 -1
  168. package/dist/assets/{timeline-definition-IT6M3QCI-hd6uJKGs.js → timeline-definition-IT6M3QCI-TKWeDnSQ.js} +1 -1
  169. package/dist/assets/{toggle-RCwU-rnE.js → toggle-C_gNjXg8.js} +1 -1
  170. package/dist/assets/{tooltip-D9723Brr.js → tooltip-CR_izE8Q.js} +1 -1
  171. package/dist/assets/tracing-DwAC7DWN.js +1 -0
  172. package/dist/assets/tracing-panel-D1iNgclL.js +2 -0
  173. package/dist/assets/{tree-Ch2-GuhG.js → tree-WVrWjdwv.js} +3 -3
  174. package/dist/assets/{useAddCell-YC7rpcmD.js → useAddCell-D6Q7JziZ.js} +1 -1
  175. package/dist/assets/{useBoolean-m1e6E3Ao.js → useBoolean-CrltYVhE.js} +1 -1
  176. package/dist/assets/{useCellActionButton-GUb2fXU8.js → useCellActionButton-B2G9OaX9.js} +1 -1
  177. package/dist/assets/{useDateFormatter-CpE7XQLs.js → useDateFormatter-DOoVZUts.js} +1 -1
  178. package/dist/assets/{useDeleteCell-DRUDRiy0.js → useDeleteCell-d8veThW7.js} +1 -1
  179. package/dist/assets/{useDependencyPanelTab-nFVMlEx0.js → useDependencyPanelTab-B3iIj3MO.js} +1 -1
  180. package/dist/assets/{useInstallPackage-C9V-on2J.js → useInstallPackage-CQEr5429.js} +1 -1
  181. package/dist/assets/{useNotebookActions-ChLHy-0O.js → useNotebookActions-SAEOWcBo.js} +1 -1
  182. package/dist/assets/{useNumberFormatter-DbDKSvEd.js → useNumberFormatter-BXZcbTzH.js} +1 -1
  183. package/dist/assets/{usePress-BHGkpw8X.js → usePress-alQ5Crny.js} +1 -1
  184. package/dist/assets/{useRunCells-5m6jCnyo.js → useRunCells-Dhl8ZTGh.js} +1 -1
  185. package/dist/assets/{useSplitCell-DGD9smMq.js → useSplitCell-BSnFazbH.js} +1 -1
  186. package/dist/assets/{utils-CdjCA1J8.js → utils-C24l2A1T.js} +1 -1
  187. package/dist/assets/{vega-component-CRbeDmeM.js → vega-component-9Pf4pVZL.js} +1 -1
  188. package/dist/assets/{vega-loader.browser-CZV6_g2i.js → vega-loader.browser-BJ9oKrvH.js} +1 -1
  189. package/dist/assets/{write-secret-modal-BCvuRAFb.js → write-secret-modal-BFCsWMoW.js} +1 -1
  190. package/dist/assets/{xychartDiagram-PRI3JC2R-CXlUBSbQ.js → xychartDiagram-PRI3JC2R-CIp-yeSa.js} +1 -1
  191. package/dist/index.html +125 -125
  192. package/package.json +1 -1
  193. package/src/__mocks__/notebook.ts +9 -9
  194. package/src/__tests__/branded.ts +20 -0
  195. package/src/components/app-config/user-config-form.tsx +5 -4
  196. package/src/components/data-table/__tests__/utils.test.ts +138 -1
  197. package/src/components/data-table/charts/__tests__/storage.test.ts +7 -7
  198. package/src/components/data-table/context-menu.tsx +9 -5
  199. package/src/components/data-table/data-table.tsx +3 -0
  200. package/src/components/data-table/range-focus/__tests__/atoms.test.ts +8 -2
  201. package/src/components/data-table/range-focus/__tests__/test-utils.ts +2 -0
  202. package/src/components/data-table/range-focus/__tests__/utils.test.ts +82 -8
  203. package/src/components/data-table/range-focus/atoms.ts +2 -2
  204. package/src/components/data-table/range-focus/utils.ts +50 -12
  205. package/src/components/data-table/types.ts +7 -0
  206. package/src/components/data-table/utils.ts +87 -0
  207. package/src/components/editor/__tests__/data-attributes.test.tsx +8 -8
  208. package/src/components/editor/ai/__tests__/completion-utils.test.ts +15 -15
  209. package/src/components/editor/navigation/__tests__/clipboard.test.ts +2 -2
  210. package/src/components/editor/navigation/__tests__/selection.test.ts +7 -6
  211. package/src/components/editor/navigation/__tests__/state.test.ts +8 -7
  212. package/src/components/editor/output/MarimoErrorOutput.tsx +7 -7
  213. package/src/components/editor/output/__tests__/traceback.test.tsx +4 -4
  214. package/src/components/editor/output/console/__tests__/ConsoleOutput.test.tsx +4 -4
  215. package/src/components/editor/renderers/vertical-layout/useFocusFirstEditor.ts +8 -1
  216. package/src/components/storage/storage-file-viewer.tsx +35 -1
  217. package/src/components/storage/storage-inspector.tsx +9 -4
  218. package/src/components/storage/storage-snippets.ts +3 -3
  219. package/src/components/tracing/tracing.tsx +3 -1
  220. package/src/components/ui/range-slider.tsx +108 -1
  221. package/src/core/ai/__tests__/staged-cells.test.ts +9 -8
  222. package/src/core/ai/context/providers/__tests__/cell-output.test.ts +31 -31
  223. package/src/core/ai/context/providers/__tests__/datasource.test.ts +3 -3
  224. package/src/core/ai/context/providers/__tests__/tables.test.ts +3 -2
  225. package/src/core/ai/context/providers/__tests__/variable.test.ts +84 -63
  226. package/src/core/ai/tools/__tests__/edit-notebook-tool.test.ts +10 -9
  227. package/src/core/ai/tools/__tests__/run-cells-tool.test.ts +6 -6
  228. package/src/core/ai/tools/edit-notebook-tool.ts +3 -3
  229. package/src/core/cells/__tests__/add-missing-import.test.ts +3 -3
  230. package/src/core/cells/__tests__/cells.test.ts +192 -135
  231. package/src/core/cells/__tests__/focus.test.ts +5 -4
  232. package/src/core/cells/__tests__/logs.test.ts +13 -12
  233. package/src/core/cells/__tests__/pending-delete-service.test.tsx +3 -3
  234. package/src/core/cells/__tests__/runs.test.ts +22 -21
  235. package/src/core/cells/__tests__/scrollCellIntoView.test.ts +8 -7
  236. package/src/core/cells/__tests__/session.test.ts +23 -22
  237. package/src/core/cells/cells.ts +1 -1
  238. package/src/core/cells/ids.ts +5 -5
  239. package/src/core/cells/logs.ts +2 -2
  240. package/src/core/cells/runs.ts +6 -8
  241. package/src/core/codemirror/__tests__/format.test.ts +34 -36
  242. package/src/core/codemirror/__tests__/setup.test.ts +2 -2
  243. package/src/core/codemirror/cells/__tests__/extensions.test.ts +114 -0
  244. package/src/core/codemirror/cells/__tests__/traceback-decorations.test.ts +33 -32
  245. package/src/core/codemirror/cells/extensions.ts +66 -23
  246. package/src/core/codemirror/completion/__tests__/keymap.test.ts +15 -35
  247. package/src/core/codemirror/completion/keymap.ts +14 -4
  248. package/src/core/codemirror/copilot/__tests__/getCodes.test.ts +12 -13
  249. package/src/core/codemirror/language/__tests__/utils.test.ts +3 -3
  250. package/src/core/codemirror/language/embedded/__tests__/embedded-python.test.ts +7 -8
  251. package/src/core/codemirror/language/languages/python.ts +4 -0
  252. package/src/core/codemirror/lsp/__tests__/notebook-lsp.test.ts +4 -3
  253. package/src/core/codemirror/lsp/notebook-lsp.ts +28 -2
  254. package/src/core/codemirror/reactive-references/__tests__/analyzer.test.ts +7 -6
  255. package/src/core/codemirror/reactive-references/analyzer.ts +2 -2
  256. package/src/core/codemirror/rtc/loro/__tests__/sync.test.ts +52 -0
  257. package/src/core/codemirror/rtc/loro/sync.ts +1 -0
  258. package/src/core/datasets/__tests__/data-source.test.ts +5 -6
  259. package/src/core/datasets/state.ts +1 -1
  260. package/src/core/errors/__tests__/errors.test.ts +2 -1
  261. package/src/core/export/__tests__/hooks.test.ts +37 -36
  262. package/src/core/islands/main.ts +2 -7
  263. package/src/core/kernel/__tests__/handlers.test.ts +5 -4
  264. package/src/core/kernel/handlers.ts +7 -4
  265. package/src/core/network/DeferredRequestRegistry.ts +2 -2
  266. package/src/core/network/__tests__/CachingRequestRegistry.test.ts +9 -10
  267. package/src/core/network/__tests__/DeferredRequestRegistry.test.ts +4 -6
  268. package/src/core/static/__tests__/virtual-file-tracker.test.ts +8 -8
  269. package/src/core/static/virtual-file-tracker.ts +1 -1
  270. package/src/core/storage/__tests__/state.test.ts +31 -21
  271. package/src/core/storage/state.ts +1 -1
  272. package/src/core/variables/__tests__/state.test.ts +6 -6
  273. package/src/core/variables/types.ts +2 -2
  274. package/src/core/wasm/__tests__/state.test.ts +8 -8
  275. package/src/core/websocket/useMarimoKernelConnection.tsx +12 -15
  276. package/src/css/md-tooltip.css +4 -39
  277. package/src/css/md.css +7 -0
  278. package/src/plugins/core/RenderHTML.tsx +17 -0
  279. package/src/plugins/core/__test__/RenderHTML.test.ts +45 -0
  280. package/src/plugins/core/sanitize-html.ts +25 -18
  281. package/src/plugins/impl/DataTablePlugin.tsx +23 -2
  282. package/src/plugins/impl/SliderPlugin.tsx +1 -3
  283. package/src/plugins/impl/__tests__/SliderPlugin.test.tsx +120 -0
  284. package/src/plugins/impl/anywidget/model.ts +1 -2
  285. package/src/stories/cell.stories.tsx +8 -8
  286. package/src/stories/layout/vertical/one-column.stories.tsx +9 -8
  287. package/src/stories/log-viewer.stories.tsx +8 -8
  288. package/src/stories/variables.stories.tsx +2 -2
  289. package/src/utils/__tests__/download.test.tsx +21 -20
  290. package/src/utils/copy.ts +18 -5
  291. package/src/utils/download.ts +4 -3
  292. package/src/utils/html-to-image.ts +6 -0
  293. package/src/utils/json/base64.ts +3 -3
  294. package/src/utils/traceback.ts +5 -3
  295. package/dist/assets/JsonOutput-DKXKGKvX.js +0 -46
  296. package/dist/assets/RenderHTML-DJ8khuob.js +0 -1
  297. package/dist/assets/ai-model-dropdown-DPTa_EpU.js +0 -5
  298. package/dist/assets/cell-editor-DX7IcqNr.js +0 -23
  299. package/dist/assets/channel-CkT8Qdo2.js +0 -1
  300. package/dist/assets/classDiagram-2ON5EDUG-BhMT_rTz.js +0 -1
  301. package/dist/assets/classDiagram-v2-WZHVMYZB-D-F4WEyb.js +0 -1
  302. package/dist/assets/copy-YwM0Pd7v.js +0 -1
  303. package/dist/assets/dist-BTqMkf4j.js +0 -1
  304. package/dist/assets/dist-DHw9sXeN.js +0 -1
  305. package/dist/assets/dist-Du0qZeXz.js +0 -1
  306. package/dist/assets/dist-a0FfbvMD.js +0 -1
  307. package/dist/assets/dist-cfkfP0ni.js +0 -1
  308. package/dist/assets/dist-pzQ9JG-p.js +0 -1
  309. package/dist/assets/file-explorer-panel-CB8vF5ob.js +0 -26
  310. package/dist/assets/home-page-itW0tRmv.js +0 -4
  311. package/dist/assets/html-to-image-BnSc-Wa0.js +0 -2
  312. package/dist/assets/index-0n92c_W7.js +0 -38
  313. package/dist/assets/mermaid-4DMBBIKO-PoHnhmy8.js +0 -1
  314. package/dist/assets/react-vega-Da-Ps9UW.js +0 -1
  315. package/dist/assets/stateDiagram-v2-4FDKWEC3-CEqeIlM0.js +0 -1
  316. package/dist/assets/stex-BBWVYm-R.js +0 -1
  317. package/dist/assets/tracing-DUVd0jtl.js +0 -1
  318. package/dist/assets/tracing-panel-CuTxPn_x.js +0 -2
  319. /package/dist/assets/{Combination-Cs9nbinQ.js → Combination-CSPK4t6z.js} +0 -0
  320. /package/dist/assets/{Deferred-BMfCOLaw.js → Deferred-CfyqLOPG.js} +0 -0
  321. /package/dist/assets/{SSRProvider-DC7ElCZZ.js → SSRProvider-CwqN9FWV.js} +0 -0
  322. /package/dist/assets/{badge-hTpPIsMT.js → badge-DImLVznf.js} +0 -0
  323. /package/dist/assets/{blob-CObhN-9g.js → blob-DgooIGjS.js} +0 -0
  324. /package/dist/assets/{bot-message-square-CK6eoGWy.js → bot-message-square-Dw41U6lL.js} +0 -0
  325. /package/dist/assets/{chart-no-axes-column-DV8gdCvH.js → chart-no-axes-column-a9XtWmzk.js} +0 -0
  326. /package/dist/assets/{check-BE0hEwVo.js → check-DZA_bRpw.js} +0 -0
  327. /package/dist/assets/{chevron-right-D0GQBpTb.js → chevron-right-CvVxySQk.js} +0 -0
  328. /package/dist/assets/{circle-check-gLIOLu8x.js → circle-check-CbaVeozR.js} +0 -0
  329. /package/dist/assets/{circle-play-DYGULlKZ.js → circle-play-DYn5nR6N.js} +0 -0
  330. /package/dist/assets/{circle-plus-CGG-gArM.js → circle-plus-nCze0-py.js} +0 -0
  331. /package/dist/assets/{clipboard-paste-DoYSN8Sv.js → clipboard-paste-EHXeKq9D.js} +0 -0
  332. /package/dist/assets/{code-xml-DwHPF_nL.js → code-xml-BlrJCgNZ.js} +0 -0
  333. /package/dist/assets/{copy-CkudG0Ej.js → copy-D6N1-xc1.js} +0 -0
  334. /package/dist/assets/{database-zap-DTWCDKdn.js → database-zap-BIGMFOfP.js} +0 -0
  335. /package/dist/assets/{defaultLocale-DK1MWd7f.js → defaultLocale-CGfP-Ye3.js} +0 -0
  336. /package/dist/assets/{defaultLocale-OkOxlkkM.js → defaultLocale-CuYNS33t.js} +0 -0
  337. /package/dist/assets/{dist-8UD0A5sU.js → dist-BF9S272t.js} +0 -0
  338. /package/dist/assets/{dist-apDpadc4.js → dist-Bk1itfBD.js} +0 -0
  339. /package/dist/assets/{dist-CQqv2gQL.js → dist-Dm11d0_A.js} +0 -0
  340. /package/dist/assets/{download-DBW9RXtT.js → download-nLboiTtW.js} +0 -0
  341. /package/dist/assets/{ellipsis-0_zJdF6H.js → ellipsis-d7eaKIFn.js} +0 -0
  342. /package/dist/assets/{ellipsis-vertical-CAB7tdza.js → ellipsis-vertical-DBQ5kWTo.js} +0 -0
  343. /package/dist/assets/{emotion-is-prop-valid.esm-D1keIaYa.js → emotion-is-prop-valid.esm-D7FeWASw.js} +0 -0
  344. /package/dist/assets/{errors-Bfogio62.js → errors-0IrrdfSG.js} +0 -0
  345. /package/dist/assets/{extends-DRbCSry7.js → extends-aq1t6BkR.js} +0 -0
  346. /package/dist/assets/{eye-off-vwi9L975.js → eye-off-CF3GmvXV.js} +0 -0
  347. /package/dist/assets/{file-DzHkbIdO.js → file-C-yMeaec.js} +0 -0
  348. /package/dist/assets/{file-headphone-B5q2Ow55.js → file-headphone-CPAP8asn.js} +0 -0
  349. /package/dist/assets/{file-plus-corner-lLQw9OnR.js → file-plus-corner-ks__N1mr.js} +0 -0
  350. /package/dist/assets/{github-BVtI-3F1.js → github-CRD4USKm.js} +0 -0
  351. /package/dist/assets/{image-DXfkah9d.js → image-CfyJzBP9.js} +0 -0
  352. /package/dist/assets/{link-Cf10mh3t.js → link-_dbp0XNB.js} +0 -0
  353. /package/dist/assets/{maps-OKerBHH8.js → maps-DQsjfyTy.js} +0 -0
  354. /package/dist/assets/{numbers-CYnquDho.js → numbers-Cno6K0UF.js} +0 -0
  355. /package/dist/assets/{objectWithoutPropertiesLoose-DP4vAkvg.js → objectWithoutPropertiesLoose-Dxmp_Bd_.js} +0 -0
  356. /package/dist/assets/{ordinal-BjO5SoTk.js → ordinal-CMAUv8ku.js} +0 -0
  357. /package/dist/assets/{paths-D2lG83Oh.js → paths-BVwhPRFT.js} +0 -0
  358. /package/dist/assets/{play-DKSqmedg.js → play-Bu_0ogGD.js} +0 -0
  359. /package/dist/assets/{plus-dVmh0yTy.js → plus-DJ99CUbx.js} +0 -0
  360. /package/dist/assets/{preload-helper-BW0IMuFq.js → preload-helper-y72bE5iF.js} +0 -0
  361. /package/dist/assets/{prop-types-RrUi-pOT.js → prop-types-DRf51_gT.js} +0 -0
  362. /package/dist/assets/{purify.es-BBn8CPhf.js → purify.es-Cf8RQecB.js} +0 -0
  363. /package/dist/assets/{range-DNqFcYmr.js → range-ClqUI25v.js} +0 -0
  364. /package/dist/assets/{refresh-ccw-DLpfIr8v.js → refresh-ccw-DQ6SJ8UC.js} +0 -0
  365. /package/dist/assets/{refresh-cw-CHAHPgkx.js → refresh-cw-Dg9tCj4k.js} +0 -0
  366. /package/dist/assets/{rotate-ccw-hLlF_82X.js → rotate-ccw-BCkZViUZ.js} +0 -0
  367. /package/dist/assets/{save-8fSvKYJT.js → save-CUdcv5qm.js} +0 -0
  368. /package/dist/assets/{session-BrEm7qNv.js → session-CByuQ-M-.js} +0 -0
  369. /package/dist/assets/{settings-CzQUw9rV.js → settings-B7nhfCat.js} +0 -0
  370. /package/dist/assets/{spinner-C5JoisA7.js → spinner-pCogyRyo.js} +0 -0
  371. /package/dist/assets/{square-CxAsQQ77.js → square-KVNDGpgy.js} +0 -0
  372. /package/dist/assets/{square-function-B006EYFX.js → square-function-BMNCw7Qb.js} +0 -0
  373. /package/dist/assets/{stex-0ac7Aukl.js → stex-C6JeW1YI.js} +0 -0
  374. /package/dist/assets/{table-BCnp9gKC.js → table-Od8PbuV-.js} +0 -0
  375. /package/dist/assets/{toDate-BqKH-Jd9.js → toDate-Cfp9W_O9.js} +0 -0
  376. /package/dist/assets/{trash-2-CAoNMkjq.js → trash-2-Bc_J7TQO.js} +0 -0
  377. /package/dist/assets/{trash-BZMAQneW.js → trash-Cxd189Vw.js} +0 -0
  378. /package/dist/assets/{triangle-alert-qZo1ox6x.js → triangle-alert-CzLrbLGN.js} +0 -0
  379. /package/dist/assets/{types-C1RN112u.js → types-CLOMZuqU.js} +0 -0
  380. /package/dist/assets/{use-toast-Hc8CXlvz.js → use-toast-BtZldTi5.js} +0 -0
  381. /package/dist/assets/{useDebounce-B0dx2Gp0.js → useDebounce-DwTO_rGp.js} +0 -0
  382. /package/dist/assets/{useTheme-BYXBU1of.js → useTheme-KDW4sktg.js} +0 -0
  383. /package/dist/assets/{uuid-ClFZlR7U.js → uuid-BukULOeS.js} +0 -0
  384. /package/dist/assets/{workflow-BmeqNuSH.js → workflow-BqHyyStM.js} +0 -0
  385. /package/dist/assets/{x-BI1M8X_v.js → x-CdLP7-v3.js} +0 -0
  386. /package/dist/assets/{youtube-DE-Ej6FR.js → youtube-CfU-SnDw.js} +0 -0
@@ -22,16 +22,16 @@ export const MockNotebook = {
22
22
  },
23
23
 
24
24
  notebookState: (opts?: {
25
- cellData: Record<string, Partial<CellData>>;
26
- cellRuntime?: Record<string, Partial<CellRuntimeState>>;
25
+ cellData: Record<CellId, Partial<CellData>>;
26
+ cellRuntime?: Record<CellId, Partial<CellRuntimeState>>;
27
27
  }): NotebookState => {
28
- const cellData = opts?.cellData || {};
29
- const cellRuntime = opts?.cellRuntime || {};
28
+ const cellData = opts?.cellData ?? {};
29
+ const cellRuntime = opts?.cellRuntime ?? {};
30
30
  return {
31
- cellData: Objects.mapValues(cellData, (data, cellId) => ({
32
- id: cellId as CellId,
31
+ cellData: Objects.mapValues(cellData, (data, cid) => ({
32
+ id: cid,
33
33
  code: "",
34
- name: `cell-${cellId}`,
34
+ name: `cell-${cid}`,
35
35
  config: {
36
36
  hide_code: false,
37
37
  disabled: false,
@@ -45,8 +45,8 @@ export const MockNotebook = {
45
45
  ...data,
46
46
  })),
47
47
  cellIds: MultiColumn.from([Object.keys(cellData) as CellId[]]),
48
- cellRuntime: Objects.mapValues(cellData, (_data, cellId) =>
49
- createCellRuntimeState({ ...cellRuntime[cellId] }),
48
+ cellRuntime: Objects.mapValues(cellData, (_data, cid) =>
49
+ createCellRuntimeState({ ...cellRuntime[cid] }),
50
50
  ),
51
51
  cellHandles: Objects.mapValues(cellData, (_data) => createRef()),
52
52
  cellLogs: [],
@@ -0,0 +1,20 @@
1
+ /* Copyright 2026 Marimo. All rights reserved. */
2
+
3
+ /**
4
+ * Test-only helpers for constructing branded ID types from plain strings.
5
+ *
6
+ * In production code, branded types flow from the API (via codegen) or
7
+ * from designated creation points (e.g. CellId.create()). Tests need to
8
+ * construct these from string literals, which requires a cast. These
9
+ * helpers centralise that cast so test files don't scatter `as CellId`
10
+ * everywhere.
11
+ */
12
+
13
+ import type { CellId, UIElementId } from "@/core/cells/ids";
14
+ import type { RequestId } from "@/core/network/DeferredRequestRegistry";
15
+ import type { VariableName } from "@/core/variables/types";
16
+
17
+ export const cellId = (s: string) => s as CellId;
18
+ export const variableName = (s: string) => s as VariableName;
19
+ export const requestId = (s: string) => s as RequestId;
20
+ export const uiElementId = (s: string) => s as UIElementId;
@@ -2,6 +2,7 @@
2
2
 
3
3
  import { zodResolver } from "@hookform/resolvers/zod";
4
4
  import { atom, useAtom, useAtomValue, useSetAtom } from "jotai";
5
+ import { merge } from "lodash-es";
5
6
  import {
6
7
  AlertTriangleIcon,
7
8
  BrainIcon,
@@ -287,10 +288,10 @@ export const UserConfigForm: React.FC = () => {
287
288
  dirtyValues.ai = setAiModels(values.ai, dirtyValues.ai);
288
289
  }
289
290
 
290
- await saveUserConfig({ config: dirtyValues }).then(() => {
291
- // Update local state with form values
292
- setConfig((prev) => ({ ...prev, ...values }));
293
- });
291
+ await saveUserConfig({ config: dirtyValues });
292
+ // Only apply the changed keys; this avoids stale request responses
293
+ // overwriting newer config changes.
294
+ setConfig((prev) => merge({}, prev, dirtyValues));
294
295
  };
295
296
  const onSubmit = useDebouncedCallback(onSubmitNotDebounced, FORM_DEBOUNCE);
296
297
 
@@ -1,7 +1,13 @@
1
1
  /* Copyright 2026 Marimo. All rights reserved. */
2
2
 
3
+ import type { Table } from "@tanstack/react-table";
3
4
  import { describe, expect, it } from "vitest";
4
- import { getPageIndexForRow } from "../utils";
5
+ import {
6
+ getClipboardContent,
7
+ getPageIndexForRow,
8
+ getRawValue,
9
+ stringifyUnknownValue,
10
+ } from "../utils";
5
11
 
6
12
  describe("getPageIndexForRow", () => {
7
13
  it("should return null when row is on current page", () => {
@@ -77,3 +83,134 @@ describe("getPageIndexForRow", () => {
77
83
  expect(getPageIndexForRow(999, 100, 10)).toBe(99);
78
84
  });
79
85
  });
86
+
87
+ describe("stringifyUnknownValue", () => {
88
+ it("should stringify primitives", () => {
89
+ expect(stringifyUnknownValue({ value: "hello" })).toBe("hello");
90
+ expect(stringifyUnknownValue({ value: 42 })).toBe("42");
91
+ expect(stringifyUnknownValue({ value: true })).toBe("true");
92
+ expect(stringifyUnknownValue({ value: null })).toBe("null");
93
+ expect(stringifyUnknownValue({ value: undefined })).toBe("undefined");
94
+ });
95
+
96
+ it("should stringify null as empty string when flag is set", () => {
97
+ expect(
98
+ stringifyUnknownValue({ value: null, nullAsEmptyString: true }),
99
+ ).toBe("");
100
+ });
101
+
102
+ it("should JSON-stringify plain objects", () => {
103
+ expect(stringifyUnknownValue({ value: { x: 1 } })).toBe('{"x":1}');
104
+ });
105
+ });
106
+
107
+ describe("getClipboardContent", () => {
108
+ it("should use rawValue for text when it differs from displayedValue", () => {
109
+ const displayed = {
110
+ _serialized_mime_bundle: {
111
+ mimetype: "text/html",
112
+ data: '<a href="https://example.com">42</a>',
113
+ },
114
+ };
115
+ const result = getClipboardContent(42, displayed);
116
+ expect(result.text).toBe("42");
117
+ expect(result.html).toBe('<a href="https://example.com">42</a>');
118
+ });
119
+
120
+ it("should strip html for text when rawValue equals displayedValue", () => {
121
+ const mimeBundle = {
122
+ _serialized_mime_bundle: {
123
+ mimetype: "text/html",
124
+ data: "<b>bold</b>",
125
+ },
126
+ };
127
+ const result = getClipboardContent(mimeBundle, mimeBundle);
128
+ expect(result.text).toBe("bold");
129
+ expect(result.html).toBe("<b>bold</b>");
130
+ });
131
+
132
+ it("should handle undefined rawValue", () => {
133
+ const displayed = {
134
+ _serialized_mime_bundle: {
135
+ mimetype: "text/html",
136
+ data: "<b>hello</b>",
137
+ },
138
+ };
139
+ const result = getClipboardContent(undefined, displayed);
140
+ expect(result.text).toBe("hello");
141
+ expect(result.html).toBe("<b>hello</b>");
142
+ });
143
+
144
+ it("should return no html for plain values", () => {
145
+ const result = getClipboardContent(undefined, "plain text");
146
+ expect(result.text).toBe("plain text");
147
+ expect(result.html).toBeUndefined();
148
+ });
149
+
150
+ it("should treat text/markdown as html since mo.md() data is rendered html", () => {
151
+ const displayed = {
152
+ _serialized_mime_bundle: {
153
+ mimetype: "text/markdown",
154
+ data: '<span class="markdown"><strong>Hello</strong></span>',
155
+ },
156
+ };
157
+ const result = getClipboardContent(undefined, displayed);
158
+ expect(result.text).toBe("Hello");
159
+ expect(result.html).toBe(
160
+ '<span class="markdown"><strong>Hello</strong></span>',
161
+ );
162
+ });
163
+
164
+ it("should return no html for non-html mime bundles", () => {
165
+ const displayed = {
166
+ _serialized_mime_bundle: {
167
+ mimetype: "text/plain",
168
+ data: "just text",
169
+ },
170
+ };
171
+ const result = getClipboardContent(undefined, displayed);
172
+ expect(result.text).toBe("just text");
173
+ expect(result.html).toBeUndefined();
174
+ });
175
+
176
+ it("should handle null rawValue as a real value", () => {
177
+ const displayed = {
178
+ _serialized_mime_bundle: {
179
+ mimetype: "text/html",
180
+ data: "<i>N/A</i>",
181
+ },
182
+ };
183
+ const result = getClipboardContent(null, displayed);
184
+ expect(result.text).toBe("null");
185
+ expect(result.html).toBe("<i>N/A</i>");
186
+ });
187
+ });
188
+
189
+ function createMockTableWithMeta<TData>(rawData?: TData[]): Table<TData> {
190
+ return {
191
+ options: {
192
+ meta: { rawData },
193
+ },
194
+ } as unknown as Table<TData>;
195
+ }
196
+
197
+ describe("getRawValue", () => {
198
+ it("should return raw value when rawData is available", () => {
199
+ const table = createMockTableWithMeta([
200
+ { a: 10, b: 20 },
201
+ { a: 30, b: 40 },
202
+ ]);
203
+ expect(getRawValue(table, 0, "a")).toBe(10);
204
+ expect(getRawValue(table, 1, "b")).toBe(40);
205
+ });
206
+
207
+ it("should return undefined when rawData is not set", () => {
208
+ const table = createMockTableWithMeta(undefined);
209
+ expect(getRawValue(table, 0, "a")).toBeUndefined();
210
+ });
211
+
212
+ it("should return undefined when row index is out of bounds", () => {
213
+ const table = createMockTableWithMeta([{ a: 1 }]);
214
+ expect(getRawValue(table, 5, "a")).toBeUndefined();
215
+ });
216
+ });
@@ -12,7 +12,7 @@ vi.mock("@/utils/storage/storage", () => ({
12
12
  },
13
13
  }));
14
14
 
15
- import type { CellId } from "@/core/cells/ids";
15
+ import { cellId } from "@/__tests__/branded";
16
16
  import { availableStorage } from "@/utils/storage/storage";
17
17
  import { ChartSchema } from "../schemas";
18
18
  import type { TabName } from "../storage";
@@ -37,7 +37,7 @@ describe("Chart Transforms Storage", () => {
37
37
 
38
38
  it("should store and retrieve tab data", () => {
39
39
  const store = getDefaultStore();
40
- const cellId = "cell-1" as CellId;
40
+ const cid = cellId("cell-1");
41
41
  const tabData = {
42
42
  tabName: "Tab 1" as TabName,
43
43
  chartType: ChartType.LINE,
@@ -51,17 +51,17 @@ describe("Chart Transforms Storage", () => {
51
51
 
52
52
  // Set the atom value
53
53
  const newMap = new Map();
54
- newMap.set(cellId, [tabData]);
54
+ newMap.set(cid, [tabData]);
55
55
  store.set(tabsStorageAtom, newMap);
56
56
 
57
57
  // Verify the value was set
58
58
  const retrievedValue = store.get(tabsStorageAtom);
59
- expect(retrievedValue.get(cellId)).toEqual([tabData]);
59
+ expect(retrievedValue.get(cid)).toEqual([tabData]);
60
60
  });
61
61
 
62
62
  it("should handle multiple tabs for the same cell", () => {
63
63
  const store = getDefaultStore();
64
- const cellId = "cell-1" as CellId;
64
+ const cid = cellId("cell-1");
65
65
  const tabData1 = {
66
66
  tabName: "Tab 1" as TabName,
67
67
  chartType: ChartType.LINE,
@@ -85,12 +85,12 @@ describe("Chart Transforms Storage", () => {
85
85
 
86
86
  // Set the atom value
87
87
  const newMap = new Map();
88
- newMap.set(cellId, [tabData1, tabData2]);
88
+ newMap.set(cid, [tabData1, tabData2]);
89
89
  store.set(tabsStorageAtom, newMap);
90
90
 
91
91
  // Verify the value was set
92
92
  const retrievedValue = store.get(tabsStorageAtom);
93
- expect(retrievedValue.get(cellId)).toEqual([tabData1, tabData2]);
93
+ expect(retrievedValue.get(cid)).toEqual([tabData1, tabData2]);
94
94
  });
95
95
  });
96
96
 
@@ -18,7 +18,7 @@ import {
18
18
  import { DATA_CELL_ID } from "./cell-utils";
19
19
  import { Filter } from "./filters";
20
20
  import { selectedCellsAtom } from "./range-focus/atoms";
21
- import { stringifyUnknownValue } from "./utils";
21
+ import { getClipboardContent, getRawValue } from "./utils";
22
22
 
23
23
  export const DataTableContextMenu = <TData,>({
24
24
  contextMenuRef,
@@ -82,11 +82,15 @@ export const CellContextMenu = <TData,>({
82
82
  return;
83
83
  }
84
84
 
85
+ const table = cell.getContext().table;
86
+ const displayedValue = cell.getValue();
87
+ const rawValue =
88
+ getRawValue(table, cell.row.index, cell.column.id) ?? displayedValue;
89
+
85
90
  const handleCopyCell = () => {
86
91
  try {
87
- const value = cell.getValue();
88
- const stringValue = stringifyUnknownValue({ value });
89
- copyToClipboard(stringValue);
92
+ const { text, html } = getClipboardContent(rawValue, displayedValue);
93
+ copyToClipboard(text, html);
90
94
  } catch (error) {
91
95
  Logger.error("Failed to copy context menu cell", error);
92
96
  }
@@ -98,7 +102,7 @@ export const CellContextMenu = <TData,>({
98
102
  const handleFilterCell = (operator: "in" | "not_in") => {
99
103
  column.setFilterValue(
100
104
  Filter.select({
101
- options: [cell.getValue()],
105
+ options: [rawValue],
102
106
  operator,
103
107
  }),
104
108
  );
@@ -52,6 +52,7 @@ interface DataTableProps<TData> extends Partial<DownloadActionProps> {
52
52
  maxHeight?: number;
53
53
  columns: ColumnDef<TData>[];
54
54
  data: TData[];
55
+ rawData?: TData[]; // raw data for filtering/copying (present only if format_mapping is provided)
55
56
  // Sorting
56
57
  manualSorting?: boolean; // server-side sorting
57
58
  sorting?: SortingState; // controlled sorting
@@ -103,6 +104,7 @@ const DataTableInternal = <TData,>({
103
104
  maxHeight,
104
105
  columns,
105
106
  data,
107
+ rawData,
106
108
  selection,
107
109
  totalColumns,
108
110
  totalRows,
@@ -197,6 +199,7 @@ const DataTableInternal = <TData,>({
197
199
  ],
198
200
  data,
199
201
  columns,
202
+ meta: { rawData },
200
203
  getCoreRowModel: getCoreRowModel(),
201
204
  // pagination
202
205
  rowCount: totalRows === "too_many" ? undefined : totalRows,
@@ -403,7 +403,10 @@ describe("cell selection atoms", () => {
403
403
  beforeEach(() => {
404
404
  // Reset mocks before each test
405
405
  vi.mocked(getCellValues).mockClear();
406
- vi.mocked(getCellValues).mockReturnValue("mocked cell values");
406
+ vi.mocked(getCellValues).mockReturnValue({
407
+ text: "mocked cell values",
408
+ html: undefined,
409
+ });
407
410
  });
408
411
 
409
412
  afterEach(() => {
@@ -424,7 +427,10 @@ describe("cell selection atoms", () => {
424
427
  });
425
428
 
426
429
  expect(getCellValues).toHaveBeenCalledWith(mockTable, selectedCells);
427
- expect(copyToClipboard).toHaveBeenCalledWith("mocked cell values");
430
+ expect(copyToClipboard).toHaveBeenCalledWith(
431
+ "mocked cell values",
432
+ undefined,
433
+ );
428
434
  expect(onCopyComplete).toHaveBeenCalledWith();
429
435
  expect(state.copiedCells).toEqual(selectedCells);
430
436
  });
@@ -56,12 +56,14 @@ export function createMockRow(
56
56
  export function createMockTable(
57
57
  rows: Row<unknown>[],
58
58
  columns: Column<unknown>[],
59
+ opts?: { rawData?: unknown[] },
59
60
  ): Table<unknown> {
60
61
  return {
61
62
  getRow: (id: string) => rows.find((row) => row.id === id),
62
63
  getRowModel: () => ({ rows }),
63
64
  getColumn: (columnId: string) => columns.find((col) => col.id === columnId),
64
65
  getAllColumns: () => columns,
66
+ options: { meta: { rawData: opts?.rawData } },
65
67
  } as unknown as Table<unknown>;
66
68
  }
67
69
 
@@ -20,7 +20,8 @@ describe("getCellValues", () => {
20
20
  it("should return empty string for empty selection", () => {
21
21
  const mockTable = createMockTable([], []);
22
22
  const result = getCellValues(mockTable, new Set());
23
- expect(result).toBe("");
23
+ expect(result.text).toBe("");
24
+ expect(result.html).toBeUndefined();
24
25
  });
25
26
 
26
27
  it("should ignore select checkbox in tables", () => {
@@ -29,7 +30,8 @@ describe("getCellValues", () => {
29
30
  const table = createMockTable([row], []);
30
31
 
31
32
  const result = getCellValues(table, new Set());
32
- expect(result).toBe("");
33
+ expect(result.text).toBe("");
34
+ expect(result.html).toBeUndefined();
33
35
  });
34
36
 
35
37
  it("should return single cell value", () => {
@@ -38,7 +40,8 @@ describe("getCellValues", () => {
38
40
  const table = createMockTable([row], []);
39
41
 
40
42
  const result = getCellValues(table, new Set(["0_0"]));
41
- expect(result).toBe("test");
43
+ expect(result.text).toBe("test");
44
+ expect(result.html).toBeUndefined();
42
45
  });
43
46
 
44
47
  it("should return multiple cells from same row separated by tabs", () => {
@@ -48,7 +51,8 @@ describe("getCellValues", () => {
48
51
  const table = createMockTable([row], []);
49
52
 
50
53
  const result = getCellValues(table, new Set(["0_0", "0_1"]));
51
- expect(result).toBe("value1\tvalue2");
54
+ expect(result.text).toBe("value1\tvalue2");
55
+ expect(result.html).toBeUndefined();
52
56
  });
53
57
 
54
58
  it("should return multiple rows separated by newlines", () => {
@@ -59,7 +63,8 @@ describe("getCellValues", () => {
59
63
  const table = createMockTable([row1, row2], []);
60
64
 
61
65
  const result = getCellValues(table, new Set(["0_0", "1_0"]));
62
- expect(result).toBe("row1\nrow2");
66
+ expect(result.text).toBe("row1\nrow2");
67
+ expect(result.html).toBeUndefined();
63
68
  });
64
69
 
65
70
  it("should handle missing rows gracefully", () => {
@@ -69,7 +74,7 @@ describe("getCellValues", () => {
69
74
 
70
75
  // Row "999" doesn't exist and is skipped
71
76
  const result = getCellValues(table, new Set(["0_0", "999_0"]));
72
- expect(result).toBe("test");
77
+ expect(result.text).toBe("test");
73
78
  });
74
79
 
75
80
  it("should include undefined for missing columns on existing rows", () => {
@@ -80,7 +85,7 @@ describe("getCellValues", () => {
80
85
 
81
86
  // Column "999" doesn't exist but row.getValue() returns undefined
82
87
  const result = getCellValues(table, new Set(["0_0", "0_1", "0_999"]));
83
- expect(result).toBe("test1\ttest2\tundefined");
88
+ expect(result.text).toBe("test1\ttest2\tundefined");
84
89
  });
85
90
 
86
91
  it("should handle complex data types", () => {
@@ -91,7 +96,76 @@ describe("getCellValues", () => {
91
96
  const table = createMockTable([row], []);
92
97
 
93
98
  const result = getCellValues(table, new Set(["0_0", "0_1", "0_2"]));
94
- expect(result).toBe('{"name":"test"}\tnull\tundefined');
99
+ expect(result.text).toBe('{"name":"test"}\tnull\tundefined');
100
+ expect(result.html).toBeUndefined();
101
+ });
102
+
103
+ it("should use raw values when rawData is available", () => {
104
+ const mimeBundle = {
105
+ _serialized_mime_bundle: {
106
+ mimetype: "text/html",
107
+ data: "<b>formatted_42</b>",
108
+ },
109
+ };
110
+ const cell1 = createMockCell("0_a", mimeBundle);
111
+ const cell2 = createMockCell("0_b", "displayed_b");
112
+ const row = createMockRow("0", [cell1, cell2]);
113
+ const table = createMockTable([row], [], {
114
+ rawData: [{ a: 42, b: "raw_b" }],
115
+ });
116
+
117
+ const result = getCellValues(table, new Set(["0_a", "0_b"]));
118
+ expect(result.text).toBe("42\traw_b");
119
+ expect(result.html).toBe(
120
+ "<table><tr><td><b>formatted_42</b></td><td>raw_b</td></tr></table>",
121
+ );
122
+ });
123
+
124
+ it("should fall back to displayed value when rawData is not set", () => {
125
+ const cell = createMockCell("0_name", "displayed");
126
+ const row = createMockRow("0", [cell]);
127
+ const table = createMockTable([row], []);
128
+
129
+ const result = getCellValues(table, new Set(["0_name"]));
130
+ expect(result.text).toBe("displayed");
131
+ expect(result.html).toBeUndefined();
132
+ });
133
+
134
+ it("should return html table when cells contain mime bundles", () => {
135
+ const mimeBundle = {
136
+ _serialized_mime_bundle: {
137
+ mimetype: "text/html",
138
+ data: '<a href="https://example.com">link</a>',
139
+ },
140
+ };
141
+ const cell1 = createMockCell("0_url", mimeBundle);
142
+ const cell2 = createMockCell("0_name", "plain text");
143
+ const row = createMockRow("0", [cell1, cell2]);
144
+ const table = createMockTable([row], []);
145
+
146
+ const result = getCellValues(table, new Set(["0_url", "0_name"]));
147
+ expect(result.text).toBe("link\tplain text");
148
+ expect(result.html).toBe(
149
+ '<table><tr><td><a href="https://example.com">link</a></td><td>plain text</td></tr></table>',
150
+ );
151
+ });
152
+
153
+ it("should escape html entities in non-mime cells within html table", () => {
154
+ const mimeBundle = {
155
+ _serialized_mime_bundle: {
156
+ mimetype: "text/html",
157
+ data: "<b>bold</b>",
158
+ },
159
+ };
160
+ const cell1 = createMockCell("0_html", mimeBundle);
161
+ const cell2 = createMockCell("0_text", "a < b & c > d");
162
+ const row = createMockRow("0", [cell1, cell2]);
163
+ const table = createMockTable([row], []);
164
+
165
+ const result = getCellValues(table, new Set(["0_html", "0_text"]));
166
+ expect(result.html).toBe(
167
+ "<table><tr><td><b>bold</b></td><td>a &lt; b &amp; c &gt; d</td></tr></table>",
168
+ );
95
169
  });
96
170
  });
97
171
 
@@ -167,8 +167,8 @@ const {
167
167
  return state;
168
168
  }
169
169
 
170
- const text = getCellValues(table, state.selectedCells);
171
- copyToClipboard(text);
170
+ const { text, html } = getCellValues(table, state.selectedCells);
171
+ copyToClipboard(text, html);
172
172
  onCopyComplete();
173
173
 
174
174
  return {
@@ -2,17 +2,25 @@
2
2
 
3
3
  import type { Table } from "@tanstack/react-table";
4
4
  import { SELECT_COLUMN_ID } from "../types";
5
- import { stringifyUnknownValue } from "../utils";
5
+ import { getClipboardContent, getRawValue } from "../utils";
6
6
  import type { SelectedCell } from "./atoms";
7
7
 
8
+ export interface CellValuesResult {
9
+ text: string;
10
+ html: string | undefined;
11
+ }
12
+
8
13
  /**
9
- * Get the values of the selected cells.
14
+ * Get the values of the selected cells, preferring raw (unformatted) values
15
+ * for plain text. If any cell contains HTML (e.g. a hyperlink), also builds
16
+ * an HTML table so rich content can be preserved on paste.
10
17
  */
11
18
  export function getCellValues<TData>(
12
19
  table: Table<TData>,
13
20
  selectedCellIds: Set<string>,
14
- ): string {
15
- const rowValues = new Map<string, string[]>();
21
+ ): CellValuesResult {
22
+ const rows = new Map<string, { text: string; html?: string }[]>();
23
+ let hasHtml = false;
16
24
 
17
25
  for (const cellId of selectedCellIds) {
18
26
  if (cellId.includes(SELECT_COLUMN_ID)) {
@@ -26,17 +34,46 @@ export function getCellValues<TData>(
26
34
  continue;
27
35
  }
28
36
 
29
- const cellValue = row.getValue(columnId);
30
- const values = rowValues.get(rowId) ?? [];
31
- values.push(stringifyUnknownValue({ value: cellValue }));
32
- rowValues.set(rowId, values);
37
+ const rawValue = getRawValue(table, row.index, columnId);
38
+ const { text, html } = getClipboardContent(
39
+ rawValue,
40
+ row.getValue(columnId),
41
+ );
42
+
43
+ if (html) {
44
+ hasHtml = true;
45
+ }
46
+ const cells = rows.get(rowId) ?? [];
47
+ cells.push({ text, html });
48
+ rows.set(rowId, cells);
49
+ }
50
+
51
+ const rowValues = [...rows.values()];
52
+ const tabSeparatedText = rowValues
53
+ .map((cells) => cells.map((c) => c.text).join("\t"))
54
+ .join("\n");
55
+
56
+ let htmlTable: string | undefined;
57
+ if (hasHtml) {
58
+ const htmlTableRows = rowValues
59
+ .map(
60
+ (cells) =>
61
+ `<tr>${cells.map((c) => `<td>${c.html ?? escapeHtml(c.text)}</td>`).join("")}</tr>`,
62
+ )
63
+ .join("");
64
+ htmlTable = `<table>${htmlTableRows}</table>`;
33
65
  }
34
66
 
35
- return getTabSeparatedValues([...rowValues.values()]);
67
+ return {
68
+ text: tabSeparatedText,
69
+ html: htmlTable,
70
+ };
36
71
  }
37
72
 
38
- export function getTabSeparatedValues(values: string[][]) {
39
- return values.map((row) => row.join("\t")).join("\n");
73
+ function escapeHtml(str: string): string {
74
+ const div = document.createElement("div");
75
+ div.textContent = str;
76
+ return div.innerHTML;
40
77
  }
41
78
 
42
79
  /**
@@ -74,7 +111,8 @@ export function getNumericValuesFromSelectedCells<TData>(
74
111
  continue;
75
112
  }
76
113
 
77
- const value = row.getValue(columnId);
114
+ const value =
115
+ getRawValue(table, row.index, columnId) ?? row.getValue(columnId);
78
116
 
79
117
  // Only accept numbers and strings
80
118
  // Skip booleans, null, etc.
@@ -1,8 +1,15 @@
1
1
  /* Copyright 2026 Marimo. All rights reserved. */
2
2
 
3
+ import type { RowData } from "@tanstack/react-table";
3
4
  import type { DataType } from "@/core/kernel/messages";
4
5
  import { Objects } from "@/utils/objects";
5
6
 
7
+ declare module "@tanstack/react-table" {
8
+ interface TableMeta<TData extends RowData> {
9
+ rawData?: TData[]; // raw data for filtering/copying (present only if format_mapping is provided)
10
+ }
11
+ }
12
+
6
13
  export type ColumnName = string;
7
14
 
8
15
  export const ColumnHeaderStatsKeys = [