@marimo-team/islands 0.23.7-dev3 → 0.23.7-dev30
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/{ConnectedDataExplorerComponent-PmilQqXR.js → ConnectedDataExplorerComponent-2lBNiUv6.js} +14 -14
- package/dist/{ErrorBoundary-Da4UeYxT.js → ErrorBoundary-D3wrPNma.js} +1 -1
- package/dist/{any-language-editor-BVR0l12r.js → any-language-editor-VWs_7v27.js} +15 -15
- package/dist/apl-Bdc61P1y.js +4 -0
- package/dist/{arc-CHF8PiiF.js → arc-DfkSnvZm.js} +2 -2
- package/dist/{architecture-7HQA4BMR-D0JB_3hE.js → architecture-7HQA4BMR-CS9jOrqM.js} +1 -1
- package/dist/{architectureDiagram-VXUJARFQ-BXQEUDtK.js → architectureDiagram-VXUJARFQ-CXVJxFhH.js} +16 -16
- package/dist/asciiarmor-DVRHDGzT.js +4 -0
- package/dist/asn1-BmuKfkfu.js +4 -0
- package/dist/assets/__vite-browser-external-CAdMKBac.js +1 -0
- package/dist/assets/worker-CpBbwbQo.js +73 -0
- package/dist/{blockDiagram-VD42YOAC-DhJe-Y9i.js → blockDiagram-VD42YOAC-DGDaxR8I.js} +11 -11
- package/dist/brainfuck-DPrRpyTV.js +4 -0
- package/dist/{button-CA5pI2YF.js → button-Dj4BTre0.js} +5 -0
- package/dist/{c4Diagram-YG6GDRKO-9dSfzOFR.js → c4Diagram-YG6GDRKO-C2hc6ne8.js} +5 -5
- package/dist/{capabilities-6laDasij.js → capabilities-C9rrYCzf.js} +1 -1
- package/dist/{channel-MqYIiKgS.js → channel-BBoIVUrJ.js} +1 -1
- package/dist/{chat-ui-B-gbqk_F.js → chat-ui-CufH8sfF.js} +626 -234
- package/dist/{check-CFM2mVDr.js → check-BcUIXnUT.js} +1 -1
- package/dist/{chunk-4BX2VUAB-BwfrWBqN.js → chunk-4BX2VUAB-CzXltWHN.js} +1 -1
- package/dist/{chunk-55IACEB6-D8THf2mi.js → chunk-55IACEB6-B-1mjMMC.js} +1 -1
- package/dist/{chunk-5FQGJX7Z-CO1e63h_.js → chunk-5FQGJX7Z-BOg95xG5.js} +2 -2
- package/dist/{chunk-ABZYJK2D-BrBb_0yY.js → chunk-ABZYJK2D-D0cLy8Bb.js} +2 -2
- package/dist/{chunk-ATLVNIR6-D-0XqNah.js → chunk-ATLVNIR6-BXsEjlHF.js} +2 -2
- package/dist/{chunk-B4BG7PRW-8iRKvugR.js → chunk-B4BG7PRW-Q1usn6T3.js} +7 -7
- package/dist/{chunk-CVBHYZKI-B6xhgaBd.js → chunk-CVBHYZKI-B_c5YBcW.js} +1 -1
- package/dist/{chunk-DI55MBZ5-C0_2D4m4.js → chunk-DI55MBZ5-D1qLYNrb.js} +6 -6
- package/dist/{chunk-EXTU4WIE-Jiw9ca1u.js → chunk-EXTU4WIE-BKNXdLmD.js} +2 -2
- package/dist/{chunk-FMBD7UC4-CHdus51S.js → chunk-FMBD7UC4-Ie8M9q0W.js} +1 -1
- package/dist/{chunk-HN2XXSSU-2Vfbq-kU.js → chunk-HN2XXSSU-E3n-Ys7Z.js} +1 -1
- package/dist/{chunk-JA3XYJ7Z-6wbaigKe.js → chunk-JA3XYJ7Z-D6c6cOBG.js} +3 -3
- package/dist/{chunk-JZLCHNYA-_rfptlUP.js → chunk-JZLCHNYA-BvsPHJmL.js} +6 -6
- package/dist/{chunk-MI3HLSF2-Do0-KRc0.js → chunk-MI3HLSF2-CUYEasXO.js} +1 -1
- package/dist/{chunk-N4CR4FBY-DIZG9dVD.js → chunk-N4CR4FBY-8ycT-O9a.js} +7 -7
- package/dist/{chunk-QN33PNHL-Cc64y40m.js → chunk-QN33PNHL-Bb-eUBW3.js} +2 -2
- package/dist/{chunk-QXUST7PY-BDG0-0Or.js → chunk-QXUST7PY-DV8yRwBd.js} +10 -10
- package/dist/{chunk-QZHKN3VN-B_Mdb8GC.js → chunk-QZHKN3VN-DRjXVwuJ.js} +1 -1
- package/dist/{chunk-S3R3BYOJ-DphMP0FA.js → chunk-S3R3BYOJ-mQeCz5CE.js} +4 -4
- package/dist/{chunk-TZMSLE5B-C9LUoYkc.js → chunk-TZMSLE5B-BqW10dHe.js} +3 -3
- package/dist/classDiagram-2ON5EDUG--Yh__LHb.js +30 -0
- package/dist/classDiagram-v2-WZHVMYZB-BC7X7Xtc.js +30 -0
- package/dist/{clike-sBZrGeF8.js → clike-DTxNUn7l.js} +1 -1
- package/dist/clojure-DEttQW5T.js +4 -0
- package/dist/cmake-jNlx_DaM.js +4 -0
- package/dist/cobol-BvvIm5MJ.js +4 -0
- package/dist/{code-block-37QAKDTI-0JNwiPGv.js → code-block-37QAKDTI-BsGy1AOJ.js} +1 -1
- package/dist/{code-visibility-Dk3LFszN.js → code-visibility-BI5muKNd.js} +573 -585
- package/dist/coffeescript-DTpBMyFU.js +4 -0
- package/dist/commonlisp-BmmUG8jb.js +4 -0
- package/dist/{copy-TGGAUEWp.js → copy-DLf4aN7I.js} +2 -2
- package/dist/{cose-bilkent-S5V4N54A-DXHZkJKX.js → cose-bilkent-S5V4N54A-kjoZoid4.js} +2 -2
- package/dist/crystal-BxyqmEWC.js +4 -0
- package/dist/css-BGoCtuG3.js +4 -0
- package/dist/cypher-BmCbdl3u.js +4 -0
- package/dist/d-CFrlbjZt.js +4 -0
- package/dist/{dagre-6UL2VRFP-tH87fkPA.js → dagre-6UL2VRFP-DRBWoQUw.js} +10 -10
- package/dist/{data-grid-overlay-editor-CWUN78-s.js → data-grid-overlay-editor-efe5ZagF.js} +2 -2
- package/dist/{diagram-PSM6KHXK-2VjPSCDn.js → diagram-PSM6KHXK-H66ATWP2.js} +18 -18
- package/dist/{diagram-QEK2KX5R-CiLmNyta.js → diagram-QEK2KX5R-DItl5Wns.js} +14 -14
- package/dist/{diagram-S2PKOQOG-Zha_1CLx.js → diagram-S2PKOQOG-CtuW_ZuL.js} +14 -14
- package/dist/diff-D6XwL6P8.js +4 -0
- package/dist/{dist-Brkazupz.js → dist--sWVZwjW.js} +1 -1
- package/dist/{dist-FN0ZA_8F.js → dist-21ButRCu.js} +1 -1
- package/dist/{dist-BetEKbPG.js → dist-B8RaFTRF.js} +1 -1
- package/dist/dist-BoHGySTM.js +5 -0
- package/dist/dist-ByAz19Qc.js +5 -0
- package/dist/dist-C93EysN4.js +5 -0
- package/dist/{dist-BHnX0ia_.js → dist-CY-lVor6.js} +1 -1
- package/dist/{dist-YP-G7W0f.js → dist-CYDuv4bR.js} +1 -1
- package/dist/{dist-CMjD5MQb.js → dist-Cfo5EE2t.js} +1 -1
- package/dist/dist-CjivSDvN.js +5 -0
- package/dist/dist-Cqwx-MH7.js +5 -0
- package/dist/{dist-ESg7xyoD.js → dist-D3ZI9nhS.js} +2 -2
- package/dist/{dist-DkC6YEo0.js → dist-DMZNjfX4.js} +1 -1
- package/dist/{dist-ChC1BhqM.js → dist-DbpcoFAV.js} +1 -1
- package/dist/dist-FUNenbiQ.js +5 -0
- package/dist/{dist-BEOU2g1b.js → dist-zhSud5X3.js} +1 -1
- package/dist/{dockerfile-COvlVLcE.js → dockerfile-twL37N91.js} +1 -1
- package/dist/dtd-Bw0lRN0-.js +4 -0
- package/dist/dylan-B55eBHTt.js +4 -0
- package/dist/ecl-C8G4p0wn.js +4 -0
- package/dist/eiffel-BmH46VJl.js +4 -0
- package/dist/elm-DhzeFqkl.js +4 -0
- package/dist/{erDiagram-Q2GNP2WA-biHZS05w.js → erDiagram-Q2GNP2WA--19X2kU5.js} +14 -14
- package/dist/erlang-CY-wdlsU.js +4 -0
- package/dist/{error-banner-DnBPzEWg.js → error-banner-CVkfBUT3.js} +2 -2
- package/dist/{esm-Dd1z1auZ.js → esm-CWp0KQeK.js} +1 -1
- package/dist/{esm-CYEyrE3Y.js → esm-DjNnlmpf.js} +96 -96
- package/dist/{extends-CzJgxo2J.js → extends-vAi97cpa.js} +4 -4
- package/dist/{factor-C2GT7jfQ.js → factor-CajWS6mS.js} +1 -1
- package/dist/factor-DWkgl0xw.js +4 -0
- package/dist/{flowDiagram-NV44I4VS-CWWlUpBR.js → flowDiagram-NV44I4VS-DQmWlo7f.js} +16 -16
- package/dist/{formats-CgaK7Gmx.js → formats-CpgZM9BM.js} +3 -3
- package/dist/forth-Cij_ie2t.js +4 -0
- package/dist/fortran-Br3X9cfm.js +4 -0
- package/dist/{ganttDiagram-JELNMOA3-D7B2c4Z9.js → ganttDiagram-JELNMOA3-BOGXJ8Lk.js} +7 -7
- package/dist/gas-DYsGcMN2.js +4 -0
- package/dist/gherkin-eVgXQ0fQ.js +4 -0
- package/dist/{gitGraph-G5XIXVHT-BdepdFa_.js → gitGraph-G5XIXVHT-DGlbae5m.js} +1 -1
- package/dist/{gitGraphDiagram-V2S2FVAM-CtLvNR1S.js → gitGraphDiagram-V2S2FVAM-DjzxfF0P.js} +14 -14
- package/dist/{glide-data-editor-CvlvtPWJ.js → glide-data-editor-BK9s_dqy.js} +13 -13
- package/dist/groovy-_NFHIXG7.js +4 -0
- package/dist/haskell-D6xNG_bH.js +4 -0
- package/dist/haxe-sU_rzAwn.js +5 -0
- package/dist/{html-to-image-hMMPiNe_.js → html-to-image-DxWM1HVj.js} +2395 -2301
- package/dist/idl-Ds_VbrUx.js +4 -0
- package/dist/{info-VBDWY6EO--JNA2rNu.js → info-VBDWY6EO-D2lvLLw5.js} +1 -1
- package/dist/{infoDiagram-HS3SLOUP-BbZyOxsP.js → infoDiagram-HS3SLOUP-ChNufFsP.js} +12 -12
- package/dist/{input-BAOe64zx.js → input-Cc1Vvw9A.js} +6 -6
- package/dist/javascript-CztfIl0i.js +4 -0
- package/dist/{journeyDiagram-XKPGCS4Q-BU2mjjzl.js → journeyDiagram-XKPGCS4Q-BO_O4Ij1.js} +6 -6
- package/dist/julia-DDv40QMV.js +4 -0
- package/dist/{kanban-definition-3W4ZIXB7-BlmczUuw.js → kanban-definition-3W4ZIXB7-CPpiiiWk.js} +11 -11
- package/dist/{katex-qPqrBHZ8.js → katex-9-9QRhxz.js} +1 -1
- package/dist/{label-BCWi-Oqu.js → label-BLqV33b1.js} +2 -2
- package/dist/{line-BWRi3U3S.js → line-C5s_12ee.js} +3 -3
- package/dist/{linear-DnHwODZa.js → linear-2NnK4cxi.js} +2 -2
- package/dist/livescript-BEOngLLc.js +4 -0
- package/dist/{loader-BvW0-YWZ.js → loader-Dr8Qem8p.js} +1 -1
- package/dist/lua-9-7BhQ4Y.js +4 -0
- package/dist/main.js +1175 -1142
- package/dist/mathematica-DDa0Pfxm.js +4 -0
- package/dist/mbox-iO03mmoE.js +4 -0
- package/dist/{mermaid-4DMBBIKO-CG1ECj5W.js → mermaid-4DMBBIKO-B7VQMwJx.js} +1 -1
- package/dist/{mermaid-CEbzCxCc.js → mermaid-DO-Daq7u.js} +46 -46
- package/dist/{mermaid-parser.core-CleJseNW.js → mermaid-parser.core-DreccfmS.js} +7 -7
- package/dist/{mhchem-BwoRNwg_.js → mhchem-yiCCuiEF.js} +1 -1
- package/dist/{mindmap-definition-VGOIOE7T-CcSYqYP9.js → mindmap-definition-VGOIOE7T-CC1_Vl0f.js} +13 -13
- package/dist/mirc-C9z5LT4X.js +4 -0
- package/dist/mllike-jGJbdm_C.js +6 -0
- package/dist/modelica-DzF7oIEL.js +4 -0
- package/dist/mscgen-DB-u125o.js +6 -0
- package/dist/mumps-CRTFHhzh.js +4 -0
- package/dist/nsis-C4NPTuox.js +4 -0
- package/dist/{nsis-B5K1qoyo.js → nsis-ClF3r5Tr.js} +1 -1
- package/dist/ntriples-BCOoGph1.js +4 -0
- package/dist/{number-overlay-editor-_GnlYFHC.js → number-overlay-editor-CpKi64Fy.js} +1 -1
- package/dist/octave-DTwNlazz.js +4 -0
- package/dist/{ordinal-2jIulmcR.js → ordinal-B43ZeR68.js} +1 -1
- package/dist/oz-DD38AzSz.js +4 -0
- package/dist/{packet-DYOGHKS2-CBxXGWNr.js → packet-DYOGHKS2-CmWtF3uO.js} +1 -1
- package/dist/pascal-BohSp9jV.js +4 -0
- package/dist/perl-f5OutoPM.js +4 -0
- package/dist/{pie-VRWISCQL-Bmdnqjip.js → pie-VRWISCQL-B6u8vus8.js} +1 -1
- package/dist/{pieDiagram-ADFJNKIX-DNyLF5H2.js → pieDiagram-ADFJNKIX-Di34MOFQ.js} +19 -19
- package/dist/pig-Dv7wSmHb.js +4 -0
- package/dist/powershell-rYgjKB39.js +4 -0
- package/dist/{process-output-Bza_GK7Q.js → process-output-DBYxXdrN.js} +30 -25
- package/dist/properties-BFUNLRDN.js +4 -0
- package/dist/protobuf-B9QJQPPv.js +4 -0
- package/dist/{pug-tjbzJCFk.js → pug-B_rby2yb.js} +1 -1
- package/dist/pug-DzvWpaMC.js +4 -0
- package/dist/puppet-B_K-n_xK.js +4 -0
- package/dist/python-CAiFcaA2.js +4 -0
- package/dist/q-cLeFIBLK.js +4 -0
- package/dist/{quadrantDiagram-AYHSOK5B-rXwjifrj.js → quadrantDiagram-AYHSOK5B-B9kVk1ny.js} +3 -3
- package/dist/r-04Y-Wco3.js +4 -0
- package/dist/{radar-ZZBFDIW7-BmCWDffL.js → radar-ZZBFDIW7-XAmXSa8s.js} +1 -1
- package/dist/{react-vega-B-rkEqtS.js → react-vega-Cavbrg4l.js} +1 -1
- package/dist/{react-vega-k9ODWPlI.js → react-vega-Dh6-UKKe.js} +13 -13
- package/dist/{requirementDiagram-UZGBJVZJ-DBdrMVbs.js → requirementDiagram-UZGBJVZJ-BxGfGYEx.js} +13 -13
- package/dist/{reveal-component-FAv_bXNy.js → reveal-component-CDtsSpH5.js} +10 -10
- package/dist/rpm-FUdrIia9.js +5 -0
- package/dist/ruby-DMjFXuEW.js +4 -0
- package/dist/{sankeyDiagram-TZEHDZUN-CxmzalGv.js → sankeyDiagram-TZEHDZUN-D09PBJ-n.js} +4 -4
- package/dist/sas-DzHZxjXK.js +4 -0
- package/dist/scheme-DxHd_Rb9.js +4 -0
- package/dist/semaphore-CNDGTzkX.js +46 -0
- package/dist/{sequenceDiagram-WL72ISMW-CVCDsJ9h.js → sequenceDiagram-WL72ISMW-t_Dpemj0.js} +7 -7
- package/dist/shell-C8Kwypgf.js +4 -0
- package/dist/sieve-DdyqOKXZ.js +4 -0
- package/dist/smalltalk-pB7X1D9y.js +4 -0
- package/dist/sparql-NhBO6oOa.js +4 -0
- package/dist/{spec-DSIuqd3f.js → spec-hVaaZsY5.js} +4 -4
- package/dist/{src-BY0BGg6V.js → src-Bf2iLOlr.js} +1 -1
- package/dist/{stateDiagram-FKZM4ZOC-D_2djEhW.js → stateDiagram-FKZM4ZOC-B18gTP_j.js} +16 -16
- package/dist/stateDiagram-v2-4FDKWEC3-B6e_t14A.js +29 -0
- package/dist/{step-DGAGWg3y.js → step-CWipAYTY.js} +1 -1
- package/dist/{strings-B_FOH6eV.js → strings-BiIhGaI8.js} +4 -4
- package/dist/style.css +1 -1
- package/dist/stylus-SfWSnzPv.js +4 -0
- package/dist/swift-jRPdq2zR.js +4 -0
- package/dist/{swiper-component-KkEVUDd3.js → swiper-component-DlD2GU2g.js} +2 -2
- package/dist/tcl-_hpTHGX3.js +4 -0
- package/dist/textile-C9h8slqH.js +4 -0
- package/dist/{time-CMdrp3hw.js → time-C1SGcFMH.js} +2 -2
- package/dist/{timeline-definition-IT6M3QCI-E4NzxCs3.js → timeline-definition-IT6M3QCI-DJnh1ks5.js} +3 -3
- package/dist/{toDate-CHtl9vts.js → toDate-CJWlVNGD.js} +3 -3
- package/dist/toml-DWvtinD4.js +4 -0
- package/dist/{tooltip-B0mtKTXm.js → tooltip-DRaMBu06.js} +3 -3
- package/dist/{treemap-GDKQZRPO-CoKHPxa7.js → treemap-GDKQZRPO-Du95DV6u.js} +1 -1
- package/dist/troff-Dwo_A0y7.js +4 -0
- package/dist/ttcn-V--CPFKq.js +4 -0
- package/dist/ttcn-cfg-CPSMchTG.js +4 -0
- package/dist/turtle-B4rPGBWu.js +4 -0
- package/dist/{types-DBtDeUKD.js → types-Dzuoc3LN.js} +1 -1
- package/dist/{useAsyncData-B6hCGywC.js → useAsyncData-C56Khv_R.js} +1 -1
- package/dist/{useDateFormatter-B3mCQMP3.js → useDateFormatter-B_9k85Ex.js} +2 -2
- package/dist/{useDeepCompareMemoize-CmwDuYUH.js → useDeepCompareMemoize-Dt98v2ua.js} +1 -1
- package/dist/{useIframeCapabilities-DbdLoEDm.js → useIframeCapabilities-BkYHTrss.js} +1 -1
- package/dist/{useLifecycle-CjMjllqy.js → useLifecycle-BF6-z62y.js} +3 -3
- package/dist/{useTheme-CByZUW0p.js → useTheme-DykuNHR2.js} +2 -2
- package/dist/vb-DaMBBd4j.js +4 -0
- package/dist/vbscript-BMJQqcE2.js +4 -0
- package/dist/{vega-component-CC8TqWWV.js → vega-component-BtvQ-Kc4.js} +26 -24
- package/dist/velocity-CGq2QRq2.js +4 -0
- package/dist/verilog-CUNo8F5u.js +4 -0
- package/dist/vhdl-CCzA0msW.js +4 -0
- package/dist/webidl-CqIMDIBL.js +4 -0
- package/dist/xquery-XC5Kbr-1.js +4 -0
- package/dist/{xychartDiagram-PRI3JC2R-CuxTvjw5.js → xychartDiagram-PRI3JC2R-Dk2d_bX0.js} +10 -10
- package/dist/yacas-CGOv7Dzy.js +4 -0
- package/dist/z80-CXhVmi-f.js +4 -0
- package/dist/{zod-BxdsqRPd.js → zod-BWkcDORu.js} +1 -1
- package/package.json +3 -3
- package/src/components/chat/chat-components.tsx +47 -0
- package/src/components/chat/chat-display.tsx +41 -7
- package/src/components/chat/chat-panel.tsx +37 -10
- package/src/components/chat/chat-utils.ts +42 -20
- package/src/components/chat/reasoning-accordion.tsx +14 -3
- package/src/components/chat/tool-call/shared.ts +13 -0
- package/src/components/chat/tool-call/tool-approval-card.tsx +62 -0
- package/src/components/chat/tool-call/tool-args.tsx +26 -0
- package/src/components/chat/tool-call/tool-call-view.tsx +99 -0
- package/src/components/chat/tool-call/tool-error-card.tsx +81 -0
- package/src/components/chat/tool-call/tool-history-row.tsx +153 -0
- package/src/components/chat/tool-call/tool-result.tsx +101 -0
- package/src/components/data-table/TableTopBar.tsx +5 -1
- package/src/components/data-table/data-table.tsx +5 -0
- package/src/components/data-table/download-policy/atoms.ts +10 -0
- package/src/components/data-table/export-actions.tsx +31 -4
- package/src/components/editor/actions/useNotebookActions.tsx +3 -1
- package/src/components/editor/app-container.tsx +7 -1
- package/src/components/editor/chrome/panels/context-aware-panel/context-aware-panel.tsx +10 -2
- package/src/components/editor/chrome/wrapper/app-chrome.tsx +1 -0
- package/src/components/editor/chrome/wrapper/footer.tsx +4 -1
- package/src/components/editor/chrome/wrapper/panels.tsx +4 -1
- package/src/components/editor/chrome/wrapper/sidebar.tsx +4 -1
- package/src/components/editor/controls/Controls.tsx +11 -3
- package/src/components/editor/file-tree/requesting-tree.tsx +27 -25
- package/src/components/editor/file-tree/upload.tsx +23 -24
- package/src/components/editor/header/__tests__/status.test.tsx +108 -0
- package/src/components/editor/header/status.tsx +44 -10
- package/src/components/editor/navigation/__tests__/clipboard.test.ts +106 -0
- package/src/components/editor/navigation/__tests__/navigation.test.ts +70 -0
- package/src/components/editor/navigation/clipboard.ts +99 -25
- package/src/components/editor/navigation/navigation.ts +15 -1
- package/src/components/editor/notebook-cell.tsx +3 -0
- package/src/components/pages/run-page.tsx +4 -1
- package/src/core/ai/tools/__tests__/registry.test.ts +10 -12
- package/src/core/ai/tools/registry.ts +9 -5
- package/src/core/cells/__tests__/cells.test.ts +187 -0
- package/src/core/cells/__tests__/pending-cut-service.test.tsx +123 -0
- package/src/core/cells/cells.ts +102 -17
- package/src/core/cells/document-changes.ts +6 -1
- package/src/core/cells/pending-cut-service.ts +55 -0
- package/src/core/cells/utils.ts +11 -0
- package/src/core/codemirror/cells/extensions.ts +10 -0
- package/src/core/codemirror/markdown/__tests__/commands.test.ts +3 -3
- package/src/core/codemirror/markdown/commands.ts +1 -2
- package/src/core/edit-app.tsx +2 -1
- package/src/core/hotkeys/hotkeys.ts +5 -0
- package/src/core/islands/worker/worker.tsx +3 -2
- package/src/core/network/requests-network.ts +21 -3
- package/src/core/network/types.ts +12 -1
- package/src/core/run-app.tsx +2 -1
- package/src/core/wasm/__tests__/utils.test.ts +34 -0
- package/src/core/wasm/bridge.ts +14 -1
- package/src/core/wasm/utils.ts +14 -0
- package/src/core/wasm/worker/bootstrap.ts +3 -2
- package/src/core/wasm/worker/worker.ts +3 -2
- package/src/core/websocket/__tests__/useMarimoKernelConnection.hook.test.tsx +155 -0
- package/src/core/websocket/__tests__/useMarimoKernelConnection.test.ts +137 -0
- package/src/core/websocket/transports/basic.ts +2 -0
- package/src/core/websocket/transports/transport.ts +1 -0
- package/src/core/websocket/useMarimoKernelConnection.tsx +130 -55
- package/src/core/websocket/useWebSocket.tsx +5 -2
- package/src/css/app/Cell.css +10 -0
- package/src/plugins/impl/DataTablePlugin.tsx +12 -0
- package/src/plugins/impl/TabsPlugin.tsx +35 -7
- package/src/plugins/impl/__tests__/TabsPlugin.test.tsx +154 -0
- package/src/plugins/impl/data-frames/DataFramePlugin.tsx +6 -0
- package/src/plugins/impl/vega/resolve-data.ts +8 -1
- package/src/utils/__tests__/id-tree.test.ts +71 -0
- package/src/utils/__tests__/semaphore.test.ts +218 -0
- package/src/utils/fileToBase64.ts +8 -7
- package/src/utils/id-tree.tsx +89 -0
- package/src/utils/semaphore.ts +88 -0
- package/dist/apl-BKoVld9y.js +0 -4
- package/dist/asciiarmor-DQrKIjoo.js +0 -4
- package/dist/asn1-BZvnj0dq.js +0 -4
- package/dist/assets/__vite-browser-external-rrUYDKRl.js +0 -1
- package/dist/assets/worker-Bfy15ViQ.js +0 -73
- package/dist/brainfuck-D558nlUv.js +0 -4
- package/dist/classDiagram-2ON5EDUG-CBHMR6ZU.js +0 -30
- package/dist/classDiagram-v2-WZHVMYZB-BsUtUGM_.js +0 -30
- package/dist/clojure-Cq8mTSrE.js +0 -4
- package/dist/cmake-D8HCovWK.js +0 -4
- package/dist/cobol-UolN-9iU.js +0 -4
- package/dist/coffeescript-VdNuWrt5.js +0 -4
- package/dist/commonlisp-ALX7fpDc.js +0 -4
- package/dist/crystal-PbyO9Q_s.js +0 -4
- package/dist/css-DFklJkr_.js +0 -4
- package/dist/cypher-BifNeYlv.js +0 -4
- package/dist/d-BA-JP4PJ.js +0 -4
- package/dist/diff-CtkDpav4.js +0 -4
- package/dist/dist-BuBwsFva.js +0 -5
- package/dist/dist-BzmEQ9u7.js +0 -5
- package/dist/dist-Cih01ssx.js +0 -5
- package/dist/dist-CqfONiY9.js +0 -5
- package/dist/dist-D0iD0Fi9.js +0 -5
- package/dist/dist-DtNLXm8d.js +0 -5
- package/dist/dtd-DW3_UFEG.js +0 -4
- package/dist/dylan-pDhodO2N.js +0 -4
- package/dist/ecl-BJT8-YD7.js +0 -4
- package/dist/eiffel-Dmns-9vS.js +0 -4
- package/dist/elm-Da4sO4Bz.js +0 -4
- package/dist/erlang-C-zBsDi7.js +0 -4
- package/dist/factor-4xPWlWB5.js +0 -4
- package/dist/forth-l-c75zSd.js +0 -4
- package/dist/fortran-DIujSODW.js +0 -4
- package/dist/gas-CXnG5g_b.js +0 -4
- package/dist/gherkin-VPeqd4-X.js +0 -4
- package/dist/groovy-CphhZQgg.js +0 -4
- package/dist/haskell-CCvlS5Iq.js +0 -4
- package/dist/haxe-C_bi66fP.js +0 -5
- package/dist/idl-1DcP4Dm8.js +0 -4
- package/dist/javascript-DUIGhBvO.js +0 -4
- package/dist/julia-Cs2G4PQi.js +0 -4
- package/dist/livescript-DMtVFaAN.js +0 -4
- package/dist/lua-BAoLtdJg.js +0 -4
- package/dist/mathematica-C_NoFtbo.js +0 -4
- package/dist/mbox-DcFJFYrH.js +0 -4
- package/dist/mirc-71dccf_u.js +0 -4
- package/dist/mllike-CWcOFVDq.js +0 -6
- package/dist/modelica-Ape2VXxx.js +0 -4
- package/dist/mscgen-Cc6TwbSN.js +0 -6
- package/dist/mumps-h-ZbdkJ9.js +0 -4
- package/dist/nsis-C0p3m7JW.js +0 -4
- package/dist/ntriples-c9lEeT5w.js +0 -4
- package/dist/octave-DzEgB_74.js +0 -4
- package/dist/oz-CAxvHkyQ.js +0 -4
- package/dist/pascal-BJzu1sgP.js +0 -4
- package/dist/perl--IrOzZ2Z.js +0 -4
- package/dist/pig-CiBKKNhC.js +0 -4
- package/dist/powershell-KY0j6Qop.js +0 -4
- package/dist/properties-BW8q3ziV.js +0 -4
- package/dist/protobuf-BGaeuTGV.js +0 -4
- package/dist/pug-DjOKK-4J.js +0 -4
- package/dist/puppet-DWm2o6zX.js +0 -4
- package/dist/python-Bp2gezZy.js +0 -4
- package/dist/q-DljPshos.js +0 -4
- package/dist/r-BajPMnEu.js +0 -4
- package/dist/rpm-BKx-ZZ62.js +0 -5
- package/dist/ruby-DJq_HNKc.js +0 -4
- package/dist/sas-WANvpcOU.js +0 -4
- package/dist/scheme-CliBbhGF.js +0 -4
- package/dist/shell-BwhrNUvM.js +0 -4
- package/dist/sieve-BIVePvMp.js +0 -4
- package/dist/smalltalk-D6G48JmY.js +0 -4
- package/dist/sparql-jjc3BmEP.js +0 -4
- package/dist/stateDiagram-v2-4FDKWEC3-Cv9Av10H.js +0 -29
- package/dist/stylus-WPBPQ4PE.js +0 -4
- package/dist/swift-O1Qy6iCm.js +0 -4
- package/dist/tcl-BAFdhvsi.js +0 -4
- package/dist/textile-DFuzhNLG.js +0 -4
- package/dist/toml-DRSTeely.js +0 -4
- package/dist/troff-B_ZjwBW0.js +0 -4
- package/dist/ttcn-CAyiB3ic.js +0 -4
- package/dist/ttcn-cfg-BS5_BGBJ.js +0 -4
- package/dist/turtle-CUBEDy3E.js +0 -4
- package/dist/vb-DY9S6-U2.js +0 -4
- package/dist/vbscript-gaHC39Jq.js +0 -4
- package/dist/velocity-TfCOtJZ_.js +0 -4
- package/dist/verilog-c2JOX8mv.js +0 -4
- package/dist/vhdl-dHBirRiO.js +0 -4
- package/dist/webidl-Bauj-i07.js +0 -4
- package/dist/xquery-CtaEAOt8.js +0 -4
- package/dist/yacas-BZ85agQP.js +0 -4
- package/dist/z80-hCgR-L4U.js +0 -4
- package/src/components/chat/tool-call-accordion.tsx +0 -247
- /package/dist/{ImageComparisonComponent-DaocPIse.js → ImageComparisonComponent-CNHIsPDj.js} +0 -0
- /package/dist/{Plot-PIeIvFnD.js → Plot-4wn-lMVn.js} +0 -0
- /package/dist/{apl-Dt8GMXYg.js → apl-BCgCq9iM.js} +0 -0
- /package/dist/{array-B-MVxRIF.js → array-tvvEqPy7.js} +0 -0
- /package/dist/{asciiarmor-CitDQ85h.js → asciiarmor-BtqU-KJQ.js} +0 -0
- /package/dist/{asn1-abrf9SMK.js → asn1-Dmb-dTMx.js} +0 -0
- /package/dist/{asterisk-BUZwqih-.js → asterisk-DaVJJDnV.js} +0 -0
- /package/dist/{brainfuck-BL-Boof0.js → brainfuck-C1HoZKlE.js} +0 -0
- /package/dist/{chunk-4F5CHEZ2-C6tO9vjs.js → chunk-4F5CHEZ2-BZq7Kom7.js} +0 -0
- /package/dist/{chunk-B2363JML-Ds8wZXyP.js → chunk-B2363JML-D9-XOau1.js} +0 -0
- /package/dist/{chunk-DR5Q36YT-CP69aZS_.js → chunk-DR5Q36YT-BflwErH1.js} +0 -0
- /package/dist/{chunk-FRFDVMJY-BgQv1HBE.js → chunk-FRFDVMJY-BSBUAX7r.js} +0 -0
- /package/dist/{chunk-PL6DKKU2-DHfTUHy8.js → chunk-PL6DKKU2-B0MTXvyc.js} +0 -0
- /package/dist/{chunk-SJTYNZTY-Diciw4sx.js → chunk-SJTYNZTY-CEG4F0pB.js} +0 -0
- /package/dist/{chunk-TQ3KTPDO-CQfP9npd.js → chunk-TQ3KTPDO-DiCtqVSi.js} +0 -0
- /package/dist/{chunk-UMXZTB3W-MSKeGL7W.js → chunk-UMXZTB3W-97iS1iEl.js} +0 -0
- /package/dist/{click-outside-container-BZgN7xS_.js → click-outside-container-BDd67_1U.js} +0 -0
- /package/dist/{clike-RWg7anhx.js → clike-CdT0yHjt.js} +0 -0
- /package/dist/{clojure-DaojKHow.js → clojure-CdyrCpUv.js} +0 -0
- /package/dist/{cmake-DN-_v0XE.js → cmake-BFlPxym7.js} +0 -0
- /package/dist/{cobol-C3VpMyux.js → cobol-CcJXewp8.js} +0 -0
- /package/dist/{coffeescript-DIkz3Tbt.js → coffeescript-DnKuIKRo.js} +0 -0
- /package/dist/{colors-Cn2p_FA3.js → colors-CQAOa8cK.js} +0 -0
- /package/dist/{common-keywords-hbLeU7VU.js → common-keywords-FBrXPTcz.js} +0 -0
- /package/dist/{commonlisp-CB1boOiP.js → commonlisp-B-kok83Z.js} +0 -0
- /package/dist/{crystal-DI2oCml6.js → crystal-FYRYjI1I.js} +0 -0
- /package/dist/{css-BdEVwQDV.js → css-B45lc2V3.js} +0 -0
- /package/dist/{cypher-BNHToqxU.js → cypher-DZMLyVY_.js} +0 -0
- /package/dist/{cytoscape.esm-WbbDoCfu.js → cytoscape.esm-ayF70frT.js} +0 -0
- /package/dist/{d-D7we7I1b.js → d-x-VVT4o9.js} +0 -0
- /package/dist/{diff-Cia6fzjN.js → diff-Dxe2mpXk.js} +0 -0
- /package/dist/{dist-BK-3fF4P.js → dist-B4LJpMEg.js} +0 -0
- /package/dist/{dist-CxdUraQr.js → dist-B507mf_I.js} +0 -0
- /package/dist/{dist-C89sHDXk.js → dist-BGdYVvOu.js} +0 -0
- /package/dist/{dist-DquyVv5H.js → dist-BNyrZfqT.js} +0 -0
- /package/dist/{dist-Zn0KNbo9.js → dist-Bc5pmZIw.js} +0 -0
- /package/dist/{dist-C-J0pt5p.js → dist-BvCfQQQE.js} +0 -0
- /package/dist/{dist-D9r7Cmw7.js → dist-C2ej4eOH.js} +0 -0
- /package/dist/{dist-HVuryI1a.js → dist-C34oIrQ9.js} +0 -0
- /package/dist/{dist-CGLzXdrt.js → dist-CDFZi-QD.js} +0 -0
- /package/dist/{dist-C9fmTOin.js → dist-CYEylvZA.js} +0 -0
- /package/dist/{dist-DadjmS-4.js → dist-DJ6zJQZ4.js} +0 -0
- /package/dist/{dist-CtCY55Jf.js → dist-Dh3wkoyH.js} +0 -0
- /package/dist/{dist-C474qFoq.js → dist-Dhk6FMb0.js} +0 -0
- /package/dist/{dist-DZjQ_MBo.js → dist-KnujRhFL.js} +0 -0
- /package/dist/{dist-CinA9Enb.js → dist-WdPUFc56.js} +0 -0
- /package/dist/{dist-DBLeRrPp.js → dist-t_qL7eB8.js} +0 -0
- /package/dist/{dist-CyFFzJTb.js → dist-usPCDYx8.js} +0 -0
- /package/dist/{dtd-H4Hubdwp.js → dtd-C9VM_Wfu.js} +0 -0
- /package/dist/{duckdb-keywords-CZ_ZTscu.js → duckdb-keywords-CvJhR_Yd.js} +0 -0
- /package/dist/{dylan-fVO6rnq3.js → dylan-DTSnEIFO.js} +0 -0
- /package/dist/{ebnf-WEXPLEWb.js → ebnf-2D4Ctp3y.js} +0 -0
- /package/dist/{ecl-B94VPjNR.js → ecl-N04ptnRK.js} +0 -0
- /package/dist/{eiffel-C_R6TusS.js → eiffel-Dd8rpqr_.js} +0 -0
- /package/dist/{elm-DzCHbO2g.js → elm-GT2E866W.js} +0 -0
- /package/dist/{erlang-BGNkx6JU.js → erlang-Cf0Bp5pY.js} +0 -0
- /package/dist/{esm-Bb_hbWan.js → esm-BaaaPNGl.js} +0 -0
- /package/dist/{fcl-B_Gv5Jfx.js → fcl-Ccj8Z5Xd.js} +0 -0
- /package/dist/{forth-Bybw0cJ7.js → forth-wd_XzGTg.js} +0 -0
- /package/dist/{fortran-C6PoCLkI.js → fortran-DcwUTZFe.js} +0 -0
- /package/dist/{gas-BBlhenj4.js → gas-DeALIER3.js} +0 -0
- /package/dist/{gherkin-NXtNG85X.js → gherkin-CKTqaJNX.js} +0 -0
- /package/dist/{groovy-BoFYK9xM.js → groovy-Bwdp_d8D.js} +0 -0
- /package/dist/{haskell-BtBdvQ1n.js → haskell-DCdCcPLK.js} +0 -0
- /package/dist/{haxe-D--o6dr0.js → haxe-DAyktQWJ.js} +0 -0
- /package/dist/{http-Dc2fv19V.js → http-_DVAYWoR.js} +0 -0
- /package/dist/{idl-AqTq5l7e.js → idl-CBuZiRYu.js} +0 -0
- /package/dist/{init-D-g0ONX1.js → init-uv0kkh4g.js} +0 -0
- /package/dist/{javascript-DvwNVye9.js → javascript-DzigE11c.js} +0 -0
- /package/dist/{julia-DoKiagZC.js → julia-PwfB-0Cm.js} +0 -0
- /package/dist/{katex-B7pMJpE0.js → katex-C_XRmjAP.js} +0 -0
- /package/dist/{livescript-DxBZMiWB.js → livescript-BJLz1EbT.js} +0 -0
- /package/dist/{lua-DmS_0NTu.js → lua-ZC-XC2jf.js} +0 -0
- /package/dist/{math-BYK36kWZ.js → math-DFcdCCU8.js} +0 -0
- /package/dist/{mathematica-ChlDFeIC.js → mathematica-DCYMx6qB.js} +0 -0
- /package/dist/{mbox-CguZuODr.js → mbox-OxMK_9XI.js} +0 -0
- /package/dist/{mirc-CFtY8dqz.js → mirc-nJVyhA0H.js} +0 -0
- /package/dist/{mllike-C0EJrEOk.js → mllike-DRO89bsU.js} +0 -0
- /package/dist/{modelica-C1kO1nfS.js → modelica-Don3E6ZD.js} +0 -0
- /package/dist/{mscgen-DEYdr7AY.js → mscgen-DJfqD3bN.js} +0 -0
- /package/dist/{mumps-B3NVJs2V.js → mumps-SjGTvDYL.js} +0 -0
- /package/dist/{nginx-ComVAAGN.js → nginx-DasThI7R.js} +0 -0
- /package/dist/{node-sql-parser-DNGGJ-Rw.js → node-sql-parser-B8nBD36q.js} +0 -0
- /package/dist/{ntriples-DHol9X9H.js → ntriples-CNBKRl3I.js} +0 -0
- /package/dist/{octave-CYGz0bfo.js → octave-DdeVHNlx.js} +0 -0
- /package/dist/{oz-kPxb2ni5.js → oz-CcKSoNvN.js} +0 -0
- /package/dist/{pascal-bZ0yrJKy.js → pascal-6leftwNj.js} +0 -0
- /package/dist/{path-Du6n3sOU.js → path-BGaWgPKg.js} +0 -0
- /package/dist/{perl-z4hvqyqz.js → perl-BhJIwWzN.js} +0 -0
- /package/dist/{pig-DZO8QDF9.js → pig-r-xDHqRf.js} +0 -0
- /package/dist/{powershell-BSuaDQEC.js → powershell-D-BELeNi.js} +0 -0
- /package/dist/{properties-BXhGLlIx.js → properties-CnuDhbll.js} +0 -0
- /package/dist/{protobuf-DM6iybWV.js → protobuf-CO8RBhvX.js} +0 -0
- /package/dist/{puppet-Bn05sQT8.js → puppet-NmXHjLy8.js} +0 -0
- /package/dist/{python-Cvnhm0g7.js → python-DAQXi720.js} +0 -0
- /package/dist/{q-B9V8hzex.js → q-DlikXfV0.js} +0 -0
- /package/dist/{r-Cf0gFqmq.js → r-CuohilwT.js} +0 -0
- /package/dist/{rpm-D-LMkTV1.js → rpm-0Pjwp0Pb.js} +0 -0
- /package/dist/{ruby-DeuPikpK.js → ruby-Dq8NJTDG.js} +0 -0
- /package/dist/{sas-C9tjgAo9.js → sas-CuwonyVP.js} +0 -0
- /package/dist/{scheme-D1_bUF0G.js → scheme-CYU-RRIf.js} +0 -0
- /package/dist/{shell-CJBmnks3.js → shell-COPmX2qE.js} +0 -0
- /package/dist/{sieve-1fSV75CF.js → sieve-B_3zyLne.js} +0 -0
- /package/dist/{simple-mode-B90Wdavj.js → simple-mode-DSBniks8.js} +0 -0
- /package/dist/{smalltalk-sZNPD0HO.js → smalltalk-DRft7iPv.js} +0 -0
- /package/dist/{solr-DTkyqJ-Z.js → solr-RZ9uTl59.js} +0 -0
- /package/dist/{sparql-oHc1nm77.js → sparql-CN6qj55H.js} +0 -0
- /package/dist/{spreadsheet-CER0raqY.js → spreadsheet-BNNUNXA2.js} +0 -0
- /package/dist/{sql-ByOoEONQ.js → sql-B4x8IkwU.js} +0 -0
- /package/dist/{stylus-KzkX6zRB.js → stylus-Bn_ZjOQ3.js} +0 -0
- /package/dist/{swift-DqVxZvKo.js → swift-BLUJhMbz.js} +0 -0
- /package/dist/{tcl-BtWSwXfA.js → tcl-C86fxecl.js} +0 -0
- /package/dist/{textile-CWDbn9Ql.js → textile-DmHh2rsK.js} +0 -0
- /package/dist/{tiddlywiki-Cr9xyOY1.js → tiddlywiki-DI0mF2WJ.js} +0 -0
- /package/dist/{tiki-D5JONyfZ.js → tiki-2HU6XLLn.js} +0 -0
- /package/dist/{timer-D7JVdX9U.js → timer-YZl28NYN.js} +0 -0
- /package/dist/{toml-BfehlgmL.js → toml-GWANRNAD.js} +0 -0
- /package/dist/{treemap-qFGzn7xk.js → treemap-D-ka1hvx.js} +0 -0
- /package/dist/{troff-BZBk6AAu.js → troff-BHTsomIy.js} +0 -0
- /package/dist/{ttcn-DVwvXg0_.js → ttcn-DQuhn5Mn.js} +0 -0
- /package/dist/{ttcn-cfg-gjbVLf1L.js → ttcn-cfg-HjFYtdB-.js} +0 -0
- /package/dist/{turtle-CgxKXorV.js → turtle-nCay33Nv.js} +0 -0
- /package/dist/{vb-B9kSwTdM.js → vb-BG-XlqqJ.js} +0 -0
- /package/dist/{vbscript-DrUKSCdb.js → vbscript-B6vyW0-D.js} +0 -0
- /package/dist/{velocity-AlMYTnMy.js → velocity-CWegueqO.js} +0 -0
- /package/dist/{verilog-DLUaM05j.js → verilog-CzSQm4cG.js} +0 -0
- /package/dist/{vhdl-DUJOtSmO.js → vhdl-DqnNVL7r.js} +0 -0
- /package/dist/{webidl-CQp4aHk_.js → webidl-DXEUpDWH.js} +0 -0
- /package/dist/{xquery-IxkjlwOD.js → xquery-Ba_NB5bD.js} +0 -0
- /package/dist/{yacas-Bnctn5w8.js → yacas-HKQU6hyk.js} +0 -0
- /package/dist/{z80-DrFwhx53.js → z80-CXkHXLdj.js} +0 -0
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
/* Copyright 2026 Marimo. All rights reserved. */
|
|
2
|
+
// @vitest-environment jsdom
|
|
3
|
+
|
|
4
|
+
import { fireEvent, render } from "@testing-library/react";
|
|
5
|
+
import { createStore, Provider as JotaiProvider } from "jotai";
|
|
6
|
+
import type React from "react";
|
|
7
|
+
import { describe, expect, it, vi } from "vitest";
|
|
8
|
+
import { TooltipProvider } from "@/components/ui/tooltip";
|
|
9
|
+
import { viewStateAtom } from "@/core/mode";
|
|
10
|
+
import {
|
|
11
|
+
type ConnectionStatus,
|
|
12
|
+
WebSocketClosedReason,
|
|
13
|
+
WebSocketState,
|
|
14
|
+
} from "@/core/websocket/types";
|
|
15
|
+
import { StatusOverlay } from "../status";
|
|
16
|
+
|
|
17
|
+
function renderOverlay(
|
|
18
|
+
connection: ConnectionStatus,
|
|
19
|
+
onReconnect?: () => void,
|
|
20
|
+
): ReturnType<typeof render> {
|
|
21
|
+
const store = createStore();
|
|
22
|
+
store.set(viewStateAtom, { mode: "edit", cellAnchor: null });
|
|
23
|
+
const wrapper: React.FC<React.PropsWithChildren> = ({ children }) => (
|
|
24
|
+
<JotaiProvider store={store}>
|
|
25
|
+
<TooltipProvider>{children}</TooltipProvider>
|
|
26
|
+
</JotaiProvider>
|
|
27
|
+
);
|
|
28
|
+
return render(
|
|
29
|
+
<StatusOverlay
|
|
30
|
+
connection={connection}
|
|
31
|
+
isRunning={false}
|
|
32
|
+
onReconnect={onReconnect}
|
|
33
|
+
/>,
|
|
34
|
+
{ wrapper },
|
|
35
|
+
);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
describe("StatusOverlay disconnect indicator", () => {
|
|
39
|
+
it("invokes onReconnect when the disconnect icon is clicked", () => {
|
|
40
|
+
const onReconnect = vi.fn();
|
|
41
|
+
const { getByTestId } = renderOverlay(
|
|
42
|
+
{
|
|
43
|
+
state: WebSocketState.CLOSED,
|
|
44
|
+
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
45
|
+
reason: "kernel not found",
|
|
46
|
+
},
|
|
47
|
+
onReconnect,
|
|
48
|
+
);
|
|
49
|
+
|
|
50
|
+
const icon = getByTestId("disconnected-indicator") as HTMLButtonElement;
|
|
51
|
+
expect(icon.tagName).toBe("BUTTON");
|
|
52
|
+
expect(icon.disabled).toBe(false);
|
|
53
|
+
expect(icon.getAttribute("aria-label")).toBe("Reconnect to app");
|
|
54
|
+
fireEvent.click(icon);
|
|
55
|
+
expect(onReconnect).toHaveBeenCalledTimes(1);
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it("renders a disabled button when no onReconnect is provided", () => {
|
|
59
|
+
const { getByTestId } = renderOverlay({
|
|
60
|
+
state: WebSocketState.CLOSED,
|
|
61
|
+
code: WebSocketClosedReason.KERNEL_DISCONNECTED,
|
|
62
|
+
reason: "kernel not found",
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
const button = getByTestId("disconnected-indicator");
|
|
66
|
+
expect((button as HTMLButtonElement).disabled).toBe(true);
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
it.each([
|
|
70
|
+
[
|
|
71
|
+
WebSocketClosedReason.MALFORMED_QUERY,
|
|
72
|
+
"the kernel did not recognize a request; please file a bug with marimo",
|
|
73
|
+
],
|
|
74
|
+
[
|
|
75
|
+
WebSocketClosedReason.KERNEL_STARTUP_ERROR,
|
|
76
|
+
"Failed to start kernel sandbox",
|
|
77
|
+
],
|
|
78
|
+
])(
|
|
79
|
+
"renders a disabled button for non-recoverable close reason %s",
|
|
80
|
+
(code, reason) => {
|
|
81
|
+
const onReconnect = vi.fn();
|
|
82
|
+
const { getByTestId } = renderOverlay(
|
|
83
|
+
{ state: WebSocketState.CLOSED, code, reason },
|
|
84
|
+
onReconnect,
|
|
85
|
+
);
|
|
86
|
+
|
|
87
|
+
const button = getByTestId("disconnected-indicator") as HTMLButtonElement;
|
|
88
|
+
expect(button.disabled).toBe(true);
|
|
89
|
+
fireEvent.click(button);
|
|
90
|
+
expect(onReconnect).not.toHaveBeenCalled();
|
|
91
|
+
},
|
|
92
|
+
);
|
|
93
|
+
|
|
94
|
+
it("does not render the disconnect icon when another tab has taken over", () => {
|
|
95
|
+
const onReconnect = vi.fn();
|
|
96
|
+
const { queryByTestId } = renderOverlay(
|
|
97
|
+
{
|
|
98
|
+
state: WebSocketState.CLOSED,
|
|
99
|
+
code: WebSocketClosedReason.ALREADY_RUNNING,
|
|
100
|
+
reason: "another browser tab is already connected to the kernel",
|
|
101
|
+
canTakeover: true,
|
|
102
|
+
},
|
|
103
|
+
onReconnect,
|
|
104
|
+
);
|
|
105
|
+
|
|
106
|
+
expect(queryByTestId("disconnected-indicator")).toBeNull();
|
|
107
|
+
});
|
|
108
|
+
});
|
|
@@ -7,16 +7,26 @@ import { Tooltip } from "@/components/ui/tooltip";
|
|
|
7
7
|
import { notebookScrollToRunning } from "@/core/cells/actions";
|
|
8
8
|
import { onlyScratchpadIsRunningAtom } from "@/core/cells/cells";
|
|
9
9
|
import { viewStateAtom } from "@/core/mode";
|
|
10
|
-
import {
|
|
10
|
+
import {
|
|
11
|
+
type ConnectionStatus,
|
|
12
|
+
WebSocketClosedReason,
|
|
13
|
+
WebSocketState,
|
|
14
|
+
} from "@/core/websocket/types";
|
|
11
15
|
import { cn } from "@/utils/cn";
|
|
12
16
|
|
|
13
17
|
export const StatusOverlay: React.FC<{
|
|
14
18
|
connection: ConnectionStatus;
|
|
15
19
|
isRunning: boolean;
|
|
16
|
-
|
|
20
|
+
onReconnect?: () => void;
|
|
21
|
+
}> = ({ connection, isRunning, onReconnect }) => {
|
|
17
22
|
const { mode } = useAtomValue(viewStateAtom);
|
|
18
23
|
const isClosed = connection.state === WebSocketState.CLOSED;
|
|
19
24
|
const isOpen = connection.state === WebSocketState.OPEN;
|
|
25
|
+
// Only KERNEL_DISCONNECTED is recoverable by a retry. Other terminal
|
|
26
|
+
// reasons (MALFORMED_QUERY, KERNEL_STARTUP_ERROR) would deterministically
|
|
27
|
+
// fail the same way; ALREADY_RUNNING is handled by `LockedIcon` below.
|
|
28
|
+
const canReconnect =
|
|
29
|
+
isClosed && connection.code === WebSocketClosedReason.KERNEL_DISCONNECTED;
|
|
20
30
|
|
|
21
31
|
return (
|
|
22
32
|
<>
|
|
@@ -28,7 +38,11 @@ export const StatusOverlay: React.FC<{
|
|
|
28
38
|
)}
|
|
29
39
|
>
|
|
30
40
|
{isOpen && isRunning && <RunningIcon />}
|
|
31
|
-
{isClosed && !connection.canTakeover &&
|
|
41
|
+
{isClosed && !connection.canTakeover && (
|
|
42
|
+
<DisconnectedIcon
|
|
43
|
+
onReconnect={canReconnect ? onReconnect : undefined}
|
|
44
|
+
/>
|
|
45
|
+
)}
|
|
32
46
|
{isClosed && connection.canTakeover && <LockedIcon />}
|
|
33
47
|
</div>
|
|
34
48
|
</>
|
|
@@ -37,13 +51,33 @@ export const StatusOverlay: React.FC<{
|
|
|
37
51
|
|
|
38
52
|
const topLeftStatus = "print:hidden pointer-events-auto hover:cursor-pointer";
|
|
39
53
|
|
|
40
|
-
const DisconnectedIcon
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
54
|
+
const DisconnectedIcon: React.FC<{ onReconnect?: () => void }> = ({
|
|
55
|
+
onReconnect,
|
|
56
|
+
}) => {
|
|
57
|
+
const disabled = !onReconnect;
|
|
58
|
+
return (
|
|
59
|
+
<Tooltip
|
|
60
|
+
content={
|
|
61
|
+
disabled ? "App disconnected" : "App disconnected — click to reconnect"
|
|
62
|
+
}
|
|
63
|
+
>
|
|
64
|
+
{/* Wrapper span keeps the tooltip reachable when the button is
|
|
65
|
+
disabled — a disabled <button> swallows pointer events. */}
|
|
66
|
+
<span tabIndex={disabled ? 0 : -1}>
|
|
67
|
+
<button
|
|
68
|
+
type="button"
|
|
69
|
+
className={cn(topLeftStatus, "bg-transparent border-0 p-0")}
|
|
70
|
+
aria-label={disabled ? "App disconnected" : "Reconnect to app"}
|
|
71
|
+
data-testid="disconnected-indicator"
|
|
72
|
+
onClick={onReconnect}
|
|
73
|
+
disabled={disabled}
|
|
74
|
+
>
|
|
75
|
+
<UnlinkIcon className="w-[25px] h-[25px] text-(--red-11)" />
|
|
76
|
+
</button>
|
|
77
|
+
</span>
|
|
78
|
+
</Tooltip>
|
|
79
|
+
);
|
|
80
|
+
};
|
|
47
81
|
|
|
48
82
|
const LockedIcon = () => (
|
|
49
83
|
<Tooltip content="Notebook locked">
|
|
@@ -20,6 +20,17 @@ vi.mock("@/utils/Logger", () => ({
|
|
|
20
20
|
Logger: Mocks.quietLogger(),
|
|
21
21
|
}));
|
|
22
22
|
|
|
23
|
+
const mockClearPendingCut = vi.hoisted(() => vi.fn());
|
|
24
|
+
vi.mock("@/core/cells/pending-cut-service", () => ({
|
|
25
|
+
usePendingCutActions: () => ({
|
|
26
|
+
markForCut: vi.fn(),
|
|
27
|
+
clear: mockClearPendingCut,
|
|
28
|
+
}),
|
|
29
|
+
usePendingCutState: () => ({
|
|
30
|
+
cellIds: new Set(),
|
|
31
|
+
}),
|
|
32
|
+
}));
|
|
33
|
+
|
|
23
34
|
import { MockNotebook } from "@/__mocks__/notebook";
|
|
24
35
|
import { toast } from "@/components/ui/use-toast";
|
|
25
36
|
import { getNotebook, useCellActions } from "@/core/cells/cells";
|
|
@@ -198,6 +209,101 @@ describe("useCellClipboard", () => {
|
|
|
198
209
|
description: "Cell has been copied to clipboard.",
|
|
199
210
|
});
|
|
200
211
|
});
|
|
212
|
+
|
|
213
|
+
it("should clear pending cut when copy succeeds", async () => {
|
|
214
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
215
|
+
|
|
216
|
+
await act(async () => {
|
|
217
|
+
await result.current.copyCells([mockCellId1]);
|
|
218
|
+
});
|
|
219
|
+
|
|
220
|
+
expect(mockClearPendingCut).toHaveBeenCalled();
|
|
221
|
+
});
|
|
222
|
+
|
|
223
|
+
it("should clear pending cut when copy falls back to writeText", async () => {
|
|
224
|
+
mockClipboard.write.mockRejectedValue(new Error("Write failed"));
|
|
225
|
+
mockClipboard.writeText.mockResolvedValue(undefined);
|
|
226
|
+
|
|
227
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
228
|
+
|
|
229
|
+
await act(async () => {
|
|
230
|
+
await result.current.copyCells([mockCellId1]);
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
expect(mockClearPendingCut).toHaveBeenCalled();
|
|
234
|
+
});
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
describe("cutCells", () => {
|
|
238
|
+
it("should cut single cell to clipboard with custom mimetype and plain text", async () => {
|
|
239
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
240
|
+
|
|
241
|
+
await act(async () => {
|
|
242
|
+
await result.current.cutCells([mockCellId1]);
|
|
243
|
+
});
|
|
244
|
+
|
|
245
|
+
expect(mockClipboard.write).toHaveBeenCalledWith([
|
|
246
|
+
expect.objectContaining({
|
|
247
|
+
types: ["web application/x-marimo-cell", "text/plain"],
|
|
248
|
+
}),
|
|
249
|
+
]);
|
|
250
|
+
});
|
|
251
|
+
|
|
252
|
+
it("should cut multiple cells to clipboard with custom mimetype and plain text", async () => {
|
|
253
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
254
|
+
|
|
255
|
+
await act(async () => {
|
|
256
|
+
await result.current.cutCells([mockCellId1, mockCellId2]);
|
|
257
|
+
});
|
|
258
|
+
|
|
259
|
+
expect(mockClipboard.write).toHaveBeenCalledWith([
|
|
260
|
+
expect.objectContaining({
|
|
261
|
+
types: ["web application/x-marimo-cell", "text/plain"],
|
|
262
|
+
}),
|
|
263
|
+
]);
|
|
264
|
+
});
|
|
265
|
+
|
|
266
|
+
it("should not write when no cells found", async () => {
|
|
267
|
+
asMock(getNotebook).mockReturnValue(MockNotebook.notebookState());
|
|
268
|
+
|
|
269
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
270
|
+
|
|
271
|
+
await act(async () => {
|
|
272
|
+
await result.current.cutCells([mockCellId1]);
|
|
273
|
+
});
|
|
274
|
+
|
|
275
|
+
expect(mockClipboard.write).not.toHaveBeenCalled();
|
|
276
|
+
});
|
|
277
|
+
|
|
278
|
+
it("should fallback to writeText when clipboard.write fails", async () => {
|
|
279
|
+
mockClipboard.write.mockRejectedValue(new Error("Write failed"));
|
|
280
|
+
mockClipboard.writeText.mockResolvedValue(undefined);
|
|
281
|
+
|
|
282
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
283
|
+
|
|
284
|
+
await act(async () => {
|
|
285
|
+
await result.current.cutCells([mockCellId1]);
|
|
286
|
+
});
|
|
287
|
+
|
|
288
|
+
expect(mockClipboard.write).toHaveBeenCalled();
|
|
289
|
+
expect(mockClipboard.writeText).toHaveBeenCalledWith(mockCellCode1);
|
|
290
|
+
});
|
|
291
|
+
|
|
292
|
+
it("should show error toast when both clipboard methods fail", async () => {
|
|
293
|
+
mockClipboard.write.mockRejectedValue(new Error("Write failed"));
|
|
294
|
+
mockClipboard.writeText.mockRejectedValue(new Error("WriteText failed"));
|
|
295
|
+
|
|
296
|
+
const { result } = renderHook(() => useCellClipboard());
|
|
297
|
+
|
|
298
|
+
await act(async () => {
|
|
299
|
+
await result.current.cutCells([mockCellId1]);
|
|
300
|
+
});
|
|
301
|
+
|
|
302
|
+
expect(Logger.error).toHaveBeenCalledWith(
|
|
303
|
+
"Failed to cut cells to clipboard",
|
|
304
|
+
expect.any(Error),
|
|
305
|
+
);
|
|
306
|
+
});
|
|
201
307
|
});
|
|
202
308
|
|
|
203
309
|
describe("pasteCell", () => {
|
|
@@ -44,6 +44,11 @@ vi.mock("../../cell/useRunCells", () => ({
|
|
|
44
44
|
useRunCells: vi.fn(),
|
|
45
45
|
}));
|
|
46
46
|
|
|
47
|
+
vi.mock("../../cell/useDeleteCell", () => ({
|
|
48
|
+
useDeleteCellCallback: vi.fn(),
|
|
49
|
+
useDeleteManyCellsCallback: vi.fn(),
|
|
50
|
+
}));
|
|
51
|
+
|
|
47
52
|
vi.mock("../clipboard", () => ({
|
|
48
53
|
useCellClipboard: vi.fn(),
|
|
49
54
|
}));
|
|
@@ -112,6 +117,7 @@ const mockSaveIfNotebookIsPersistent = vi.fn();
|
|
|
112
117
|
const mockSaveNotebook = vi.fn();
|
|
113
118
|
const mockRunCell = vi.fn();
|
|
114
119
|
const mockCopyCell = vi.fn();
|
|
120
|
+
const mockCutCell = vi.fn().mockResolvedValue(undefined);
|
|
115
121
|
const mockPasteCell = vi.fn();
|
|
116
122
|
|
|
117
123
|
const mockCellActions = MockNotebook.cellActions({
|
|
@@ -165,7 +171,9 @@ describe("useCellNavigationProps", () => {
|
|
|
165
171
|
mockUseRunCells.mockReturnValue(mockRunCell);
|
|
166
172
|
mockUseCellClipboard.mockReturnValue({
|
|
167
173
|
copyCells: mockCopyCell,
|
|
174
|
+
cutCells: mockCutCell,
|
|
168
175
|
pasteAtCell: mockPasteCell,
|
|
176
|
+
clearPendingCut: vi.fn(),
|
|
169
177
|
});
|
|
170
178
|
|
|
171
179
|
// Setup default config in store
|
|
@@ -237,6 +245,45 @@ describe("useCellNavigationProps", () => {
|
|
|
237
245
|
expect(mockEvent.preventDefault).toHaveBeenCalled();
|
|
238
246
|
});
|
|
239
247
|
|
|
248
|
+
it("should cut cell when 'x' key is pressed", async () => {
|
|
249
|
+
const { result } = renderWithProvider(() =>
|
|
250
|
+
useCellNavigationProps(mockCellId, options),
|
|
251
|
+
);
|
|
252
|
+
|
|
253
|
+
const mockEvent = Mocks.keyboardEvent({ key: "x" });
|
|
254
|
+
|
|
255
|
+
await act(async () => {
|
|
256
|
+
result.current.onKeyDown?.(mockEvent);
|
|
257
|
+
await Promise.resolve();
|
|
258
|
+
});
|
|
259
|
+
|
|
260
|
+
expect(mockCutCell).toHaveBeenCalledWith([mockCellId]);
|
|
261
|
+
expect(mockEvent.preventDefault).toHaveBeenCalled();
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
it("should cut multiple selected cells when 'x' key is pressed", async () => {
|
|
265
|
+
const selectionActions = setupSelection();
|
|
266
|
+
selectionActions.select({ cellId: cellId1 });
|
|
267
|
+
selectionActions.extend({
|
|
268
|
+
cellId: cellId2,
|
|
269
|
+
allCellIds: store.get(notebookAtom).cellIds,
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
const { result } = renderWithProvider(() =>
|
|
273
|
+
useCellNavigationProps(cellId1, options),
|
|
274
|
+
);
|
|
275
|
+
|
|
276
|
+
const mockEvent = Mocks.keyboardEvent({ key: "x" });
|
|
277
|
+
|
|
278
|
+
await act(async () => {
|
|
279
|
+
result.current.onKeyDown?.(mockEvent);
|
|
280
|
+
await Promise.resolve();
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
expect(mockCutCell).toHaveBeenCalledWith([cellId1, cellId2]);
|
|
284
|
+
expect(mockEvent.preventDefault).toHaveBeenCalled();
|
|
285
|
+
});
|
|
286
|
+
|
|
240
287
|
it("should paste cell when 'v' key is pressed", () => {
|
|
241
288
|
const { result } = renderWithProvider(() =>
|
|
242
289
|
useCellNavigationProps(mockCellId, options),
|
|
@@ -775,6 +822,29 @@ describe("useCellNavigationProps", () => {
|
|
|
775
822
|
expect(mockEvent.preventDefault).toHaveBeenCalled();
|
|
776
823
|
});
|
|
777
824
|
|
|
825
|
+
it("should cut multiple cells when multiple cells selected", () => {
|
|
826
|
+
// Set up selection of multiple cells
|
|
827
|
+
const selectionActions = setupSelection();
|
|
828
|
+
selectionActions.select({ cellId: cellId1 });
|
|
829
|
+
selectionActions.extend({
|
|
830
|
+
cellId: cellId3,
|
|
831
|
+
allCellIds: store.get(notebookAtom).cellIds,
|
|
832
|
+
});
|
|
833
|
+
|
|
834
|
+
const { result } = renderWithProvider(() =>
|
|
835
|
+
useCellNavigationProps(cellId2, options),
|
|
836
|
+
);
|
|
837
|
+
|
|
838
|
+
const mockEvent = Mocks.keyboardEvent({ key: "x" });
|
|
839
|
+
|
|
840
|
+
act(() => {
|
|
841
|
+
result.current.onKeyDown?.(mockEvent);
|
|
842
|
+
});
|
|
843
|
+
|
|
844
|
+
expect(mockCutCell).toHaveBeenCalledWith([cellId1, cellId2, cellId3]);
|
|
845
|
+
expect(mockEvent.preventDefault).toHaveBeenCalled();
|
|
846
|
+
});
|
|
847
|
+
|
|
778
848
|
it("should move multiple cells up when multiple cells selected", () => {
|
|
779
849
|
// Set up selection of multiple cells
|
|
780
850
|
const selectionActions = setupSelection();
|
|
@@ -5,15 +5,22 @@ import { z } from "zod";
|
|
|
5
5
|
import { toast } from "@/components/ui/use-toast";
|
|
6
6
|
import { getNotebook, useCellActions } from "@/core/cells/cells";
|
|
7
7
|
import type { CellId } from "@/core/cells/ids";
|
|
8
|
+
import {
|
|
9
|
+
usePendingCutActions,
|
|
10
|
+
usePendingCutState,
|
|
11
|
+
} from "@/core/cells/pending-cut-service";
|
|
12
|
+
import type { CellConfig } from "@/core/network/types";
|
|
8
13
|
import { copyToClipboard } from "@/utils/copy";
|
|
9
14
|
import { Logger } from "@/utils/Logger";
|
|
10
15
|
|
|
11
16
|
// According to MDN, custom mimetypes should start with "web "
|
|
12
17
|
const MARIMO_CELL_MIMETYPE = "web application/x-marimo-cell";
|
|
13
18
|
|
|
14
|
-
interface ClipboardCellData {
|
|
19
|
+
export interface ClipboardCellData {
|
|
15
20
|
cells: {
|
|
16
21
|
code: string;
|
|
22
|
+
name?: string;
|
|
23
|
+
config?: CellConfig;
|
|
17
24
|
}[];
|
|
18
25
|
version: "1.0";
|
|
19
26
|
}
|
|
@@ -22,19 +29,51 @@ const ClipboardCellDataSchema = z.object({
|
|
|
22
29
|
cells: z.array(
|
|
23
30
|
z.object({
|
|
24
31
|
code: z.string(),
|
|
32
|
+
name: z.string().optional(),
|
|
33
|
+
config: z
|
|
34
|
+
.object({
|
|
35
|
+
column: z.union([z.number(), z.null()]).optional(),
|
|
36
|
+
disabled: z.boolean().optional(),
|
|
37
|
+
hide_code: z.boolean().optional(),
|
|
38
|
+
})
|
|
39
|
+
.optional(),
|
|
25
40
|
}),
|
|
26
41
|
),
|
|
27
42
|
version: z.literal("1.0"),
|
|
28
43
|
});
|
|
29
44
|
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
45
|
+
interface ClipboardCellInput {
|
|
46
|
+
code: string;
|
|
47
|
+
name?: string;
|
|
48
|
+
config?: CellConfig;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
function toPlainText(cells: ClipboardCellInput[]): string {
|
|
52
|
+
return cells.map((cell) => cell.code).join("\n\n");
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
async function writeCellsToClipboard(
|
|
56
|
+
cells: ClipboardCellInput[],
|
|
57
|
+
): Promise<void> {
|
|
58
|
+
const clipboardData: ClipboardCellData = {
|
|
59
|
+
cells: cells.map((cell) => ({
|
|
60
|
+
code: cell.code,
|
|
61
|
+
name: cell.name,
|
|
62
|
+
config: cell.config,
|
|
63
|
+
})),
|
|
64
|
+
version: "1.0",
|
|
65
|
+
};
|
|
66
|
+
const clipboardItem = new ClipboardItemBuilder()
|
|
67
|
+
.add(MARIMO_CELL_MIMETYPE, clipboardData)
|
|
68
|
+
.add("text/plain", toPlainText(cells))
|
|
69
|
+
.build();
|
|
70
|
+
await navigator.clipboard.write([clipboardItem]);
|
|
71
|
+
}
|
|
35
72
|
|
|
36
73
|
export function useCellClipboard() {
|
|
37
74
|
const actions = useCellActions();
|
|
75
|
+
const pendingCutActions = usePendingCutActions();
|
|
76
|
+
const pendingCutState = usePendingCutState();
|
|
38
77
|
|
|
39
78
|
const copyCells = useEvent(async (cellIds: CellId[]) => {
|
|
40
79
|
const notebook = getNotebook();
|
|
@@ -43,35 +82,20 @@ export function useCellClipboard() {
|
|
|
43
82
|
.filter(Boolean);
|
|
44
83
|
|
|
45
84
|
if (cells.length === 0) {
|
|
46
|
-
// No cells to copy
|
|
47
85
|
return;
|
|
48
86
|
}
|
|
49
87
|
|
|
50
88
|
try {
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
version: "1.0",
|
|
54
|
-
};
|
|
55
|
-
|
|
56
|
-
// Create plain text representation (joined by newlines)
|
|
57
|
-
const plainText = cells.map((cell) => cell.code).join("\n\n");
|
|
58
|
-
|
|
59
|
-
// Create clipboard item with both custom mimetype and plain text
|
|
60
|
-
const clipboardItem = new ClipboardItemBuilder()
|
|
61
|
-
.add(MARIMO_CELL_MIMETYPE, clipboardData)
|
|
62
|
-
.add("text/plain", plainText)
|
|
63
|
-
.build();
|
|
64
|
-
|
|
65
|
-
await navigator.clipboard.write([clipboardItem]);
|
|
66
|
-
|
|
89
|
+
await writeCellsToClipboard(cells);
|
|
90
|
+
pendingCutActions.clear();
|
|
67
91
|
toastSuccess(cells.length);
|
|
68
92
|
} catch (error) {
|
|
69
93
|
Logger.error("Failed to copy cells to clipboard", error);
|
|
70
94
|
|
|
71
95
|
// Fallback to simple text copy
|
|
72
96
|
try {
|
|
73
|
-
|
|
74
|
-
|
|
97
|
+
await copyToClipboard(toPlainText(cells));
|
|
98
|
+
pendingCutActions.clear();
|
|
75
99
|
toastSuccess(cells.length);
|
|
76
100
|
} catch {
|
|
77
101
|
toastError();
|
|
@@ -79,12 +103,57 @@ export function useCellClipboard() {
|
|
|
79
103
|
}
|
|
80
104
|
});
|
|
81
105
|
|
|
106
|
+
const cutCells = useEvent(async (cellIds: CellId[]) => {
|
|
107
|
+
const notebook = getNotebook();
|
|
108
|
+
const validCellIds = cellIds.filter((cellId) => notebook.cellData[cellId]);
|
|
109
|
+
const cells = validCellIds.map((cellId) => notebook.cellData[cellId]);
|
|
110
|
+
|
|
111
|
+
if (cells.length === 0) {
|
|
112
|
+
return;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
try {
|
|
116
|
+
await writeCellsToClipboard(cells);
|
|
117
|
+
pendingCutActions.markForCut({ cellIds: validCellIds });
|
|
118
|
+
} catch (error) {
|
|
119
|
+
Logger.error("Failed to cut cells to clipboard", error);
|
|
120
|
+
try {
|
|
121
|
+
await copyToClipboard(toPlainText(cells));
|
|
122
|
+
// Mark cells as pending cut instead of deleting immediately
|
|
123
|
+
pendingCutActions.markForCut({ cellIds: validCellIds });
|
|
124
|
+
} catch {
|
|
125
|
+
toastError();
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
});
|
|
129
|
+
|
|
82
130
|
interface PasteOptions {
|
|
83
131
|
before?: boolean;
|
|
84
132
|
}
|
|
85
133
|
|
|
86
134
|
const pasteAtCell = useEvent(async (cellId: CellId, opts?: PasteOptions) => {
|
|
87
135
|
const { before = false } = opts ?? {};
|
|
136
|
+
|
|
137
|
+
// Check if we have pending cut cells (internal move)
|
|
138
|
+
if (pendingCutState.cellIds.size > 0) {
|
|
139
|
+
const pendingCellIds = [...pendingCutState.cellIds];
|
|
140
|
+
const notebook = getNotebook();
|
|
141
|
+
const previousPlacements = pendingCellIds.map((id) => {
|
|
142
|
+
const column = notebook.cellIds.findWithId(id);
|
|
143
|
+
return { columnId: column.id, index: column.indexOfOrThrow(id) };
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
actions.moveCellsRelativeTo({
|
|
147
|
+
cellIds: pendingCellIds,
|
|
148
|
+
targetCellId: cellId,
|
|
149
|
+
position: before ? "before" : "after",
|
|
150
|
+
previousPlacements,
|
|
151
|
+
});
|
|
152
|
+
|
|
153
|
+
pendingCutActions.clear();
|
|
154
|
+
return;
|
|
155
|
+
}
|
|
156
|
+
|
|
88
157
|
try {
|
|
89
158
|
const clipboardItems = await navigator.clipboard.read();
|
|
90
159
|
|
|
@@ -112,6 +181,9 @@ export function useCellClipboard() {
|
|
|
112
181
|
cellId: currentCellId,
|
|
113
182
|
before,
|
|
114
183
|
code: cell.code,
|
|
184
|
+
name: cell.name,
|
|
185
|
+
config: cell.config,
|
|
186
|
+
hideCode: cell.config?.hide_code,
|
|
115
187
|
autoFocus: true,
|
|
116
188
|
});
|
|
117
189
|
}
|
|
@@ -143,7 +215,9 @@ export function useCellClipboard() {
|
|
|
143
215
|
|
|
144
216
|
return {
|
|
145
217
|
copyCells,
|
|
218
|
+
cutCells,
|
|
146
219
|
pasteAtCell,
|
|
220
|
+
clearPendingCut: pendingCutActions.clear,
|
|
147
221
|
};
|
|
148
222
|
}
|
|
149
223
|
|
|
@@ -16,6 +16,10 @@ import { cellIdsAtom, notebookAtom, useCellActions } from "@/core/cells/cells";
|
|
|
16
16
|
import { useCellFocusActions } from "@/core/cells/focus";
|
|
17
17
|
import type { CellId } from "@/core/cells/ids";
|
|
18
18
|
import { HTMLCellId } from "@/core/cells/ids";
|
|
19
|
+
import {
|
|
20
|
+
clearPendingCutAtom,
|
|
21
|
+
pendingCutCellIdsAtom,
|
|
22
|
+
} from "@/core/cells/pending-cut-service";
|
|
19
23
|
import { usePendingDeleteService } from "@/core/cells/pending-delete-service";
|
|
20
24
|
import { scrollCellIntoView } from "@/core/cells/scrollCellIntoView";
|
|
21
25
|
import {
|
|
@@ -185,7 +189,7 @@ export function useCellNavigationProps(
|
|
|
185
189
|
const temporarilyShownCodeActions = useTemporarilyShownCodeActions();
|
|
186
190
|
const runCells = useRunCells();
|
|
187
191
|
const keymapPreset = useAtomValue(keymapPresetAtom);
|
|
188
|
-
const { copyCells, pasteAtCell } = useCellClipboard();
|
|
192
|
+
const { copyCells, pasteAtCell, cutCells } = useCellClipboard();
|
|
189
193
|
const rawSelectionActions = useCellSelectionActions();
|
|
190
194
|
const isSelected = useIsCellSelected(cellId);
|
|
191
195
|
const pendingDeleteService = usePendingDeleteService();
|
|
@@ -317,6 +321,12 @@ export function useCellNavigationProps(
|
|
|
317
321
|
},
|
|
318
322
|
// Clear selection
|
|
319
323
|
Escape: () => {
|
|
324
|
+
// Clear pending cut state if any
|
|
325
|
+
const pendingCutCellIds = store.get(pendingCutCellIdsAtom);
|
|
326
|
+
if (pendingCutCellIds.size > 0) {
|
|
327
|
+
store.set(clearPendingCutAtom);
|
|
328
|
+
return true;
|
|
329
|
+
}
|
|
320
330
|
if (isSelected) {
|
|
321
331
|
selectionActions.clear();
|
|
322
332
|
return true;
|
|
@@ -510,6 +520,10 @@ export function useCellNavigationProps(
|
|
|
510
520
|
copyCells(cellIds);
|
|
511
521
|
return true;
|
|
512
522
|
}),
|
|
523
|
+
"command.cutCell": addSingleHandler((cellIds) => {
|
|
524
|
+
cutCells(cellIds);
|
|
525
|
+
return true;
|
|
526
|
+
}),
|
|
513
527
|
"command.pasteCell": (cellIds) => {
|
|
514
528
|
pasteAtCell(cellIds);
|
|
515
529
|
return true;
|
|
@@ -27,6 +27,7 @@ import { Tooltip, TooltipProvider } from "@/components/ui/tooltip";
|
|
|
27
27
|
import { aiCompletionCellAtom } from "@/core/ai/state";
|
|
28
28
|
import { outputIsLoading, outputIsStale } from "@/core/cells/cell";
|
|
29
29
|
import { isOutputEmpty } from "@/core/cells/outputs";
|
|
30
|
+
import { useIsPendingCut } from "@/core/cells/pending-cut-service";
|
|
30
31
|
import { autocompletionKeymap } from "@/core/codemirror/cm";
|
|
31
32
|
import type { LanguageAdapterType } from "@/core/codemirror/language/types";
|
|
32
33
|
import { CSSClasses } from "@/core/constants";
|
|
@@ -391,6 +392,7 @@ const EditableCellComponent = ({
|
|
|
391
392
|
const deleteCell = useDeleteCellCallback();
|
|
392
393
|
const runCell = useRunCell(cellId);
|
|
393
394
|
const { sendStdin } = useRequestClient();
|
|
395
|
+
const isPendingCut = useIsPendingCut(cellId);
|
|
394
396
|
|
|
395
397
|
const [languageAdapter, setLanguageAdapter] = useState<LanguageAdapterType>();
|
|
396
398
|
|
|
@@ -545,6 +547,7 @@ const EditableCellComponent = ({
|
|
|
545
547
|
}),
|
|
546
548
|
borderless:
|
|
547
549
|
isMarkdownCodeHidden && hasOutput && !navigationProps["data-selected"],
|
|
550
|
+
"pending-cut": isPendingCut,
|
|
548
551
|
});
|
|
549
552
|
|
|
550
553
|
const handleRefactorWithAI: OnRefactorWithAI = useEvent(
|
|
@@ -34,7 +34,10 @@ const RunPage = (props: Props) => {
|
|
|
34
34
|
|
|
35
35
|
const Watermark = () => {
|
|
36
36
|
return (
|
|
37
|
-
<div
|
|
37
|
+
<div
|
|
38
|
+
className="fixed bottom-0 right-0 z-50 print:hidden"
|
|
39
|
+
data-testid="watermark"
|
|
40
|
+
>
|
|
38
41
|
<a
|
|
39
42
|
href={Constants.githubPage}
|
|
40
43
|
target="_blank"
|