@marimo-team/frontend 0.23.9-dev34 → 0.23.9-dev36

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 (102) hide show
  1. package/dist/assets/{CellStatus-Dm3YCiM6.js → CellStatus-D_6Kac7g.js} +1 -1
  2. package/dist/assets/{ConnectedDataExplorerComponent-CduY7ZLg.js → ConnectedDataExplorerComponent-DJhqEvUc.js} +1 -1
  3. package/dist/assets/{JsonOutput-DzATTBoP.js → JsonOutput-H6HB87kK.js} +12 -12
  4. package/dist/assets/{MarimoErrorOutput-XPsNfvEO.js → MarimoErrorOutput-d8qLIXHx.js} +2 -2
  5. package/dist/assets/{RenderHTML-BvW-obUg.js → RenderHTML-CBsvWTjU.js} +1 -1
  6. package/dist/assets/{RunButton-BUq1ZIzv.js → RunButton-BpQQEGB4.js} +1 -1
  7. package/dist/assets/{add-cell-with-ai-C4H664zc.js → add-cell-with-ai-Df5gxolt.js} +1 -1
  8. package/dist/assets/{add-connection-dialog-DdJBM1-Q.js → add-connection-dialog-gM0QtdYy.js} +1 -1
  9. package/dist/assets/{agent-panel-BB1ZRQse.js → agent-panel-CJcG1enH.js} +1 -1
  10. package/dist/assets/{ai-model-dropdown-9UrkfG-1.js → ai-model-dropdown-M1IdPOVK.js} +4 -4
  11. package/dist/assets/{app-config-button-N_G5k5XW.js → app-config-button-CiN27Nmp.js} +1 -1
  12. package/dist/assets/cell-editor-CNmFerQl.js +20 -0
  13. package/dist/assets/{cell-link-Bkd6QnOS.js → cell-link-BEtbRN19.js} +1 -1
  14. package/dist/assets/{cells-BP6kg4iF.js → cells-BXZ3FSp9.js} +2 -2
  15. package/dist/assets/{chat-display-76m4LMD2.js → chat-display-BFLNH0B7.js} +1 -1
  16. package/dist/assets/{chat-panel-BAIgHYyO.js → chat-panel-C2Ky6IBI.js} +1 -1
  17. package/dist/assets/{chat-ui-BggP20am.js → chat-ui-B7g-u_qk.js} +1 -1
  18. package/dist/assets/{column-preview-9_vf_vQt.js → column-preview-DDR-SBCD.js} +1 -1
  19. package/dist/assets/{command-palette-drFTNlvg.js → command-palette-C0bO32IA.js} +1 -1
  20. package/dist/assets/{common-G2mwxgr_.js → common-DeA5uNL9.js} +1 -1
  21. package/dist/assets/{components-D2tmc0KN.js → components-Cng4XB_w.js} +1 -1
  22. package/dist/assets/{components-DvggA56E.js → components-CpiEO4PD.js} +1 -1
  23. package/dist/assets/{config-hed0CaQL.js → config-BG-ACF7_.js} +1 -1
  24. package/dist/assets/{datasource-BjfZ5rCV.js → datasource-WRQecO8l.js} +1 -1
  25. package/dist/assets/{dependency-graph-panel-BQlFSqu5.js → dependency-graph-panel-DxriiKwj.js} +1 -1
  26. package/dist/assets/{documentation-panel-BbVMQHz5.js → documentation-panel-CWdrQzu5.js} +1 -1
  27. package/dist/assets/{download-DrC3sFE3.js → download-96wP8V0p.js} +1 -1
  28. package/dist/assets/{edit-page-ClIQhScA.js → edit-page-CvlbW4P0.js} +6 -6
  29. package/dist/assets/{error-panel-aaHrLkBh.js → error-panel-8XzSuC1I.js} +1 -1
  30. package/dist/assets/{file-explorer-panel-BQ9nMxks.js → file-explorer-panel-a5ksD489.js} +3 -3
  31. package/dist/assets/{file-icons-pUZ3DUy5.js → file-icons-CVihyOvq.js} +1 -1
  32. package/dist/assets/{floating-outline-CDjyi71K.js → floating-outline-Cvm2qN8K.js} +1 -1
  33. package/dist/assets/{focus-D2kXD_rc.js → focus-DDYdCBX6.js} +1 -1
  34. package/dist/assets/{form-COQrUWdR.js → form-DLzm3BCq.js} +1 -1
  35. package/dist/assets/{glide-data-editor-7_pHQtPX.js → glide-data-editor-Dg0e0Gop.js} +1 -1
  36. package/dist/assets/{globals-Dd7eo1nD.js → globals-CYbHM4QS.js} +1 -1
  37. package/dist/assets/{home-page-DNFOElFw.js → home-page-uHyo5m6a.js} +1 -1
  38. package/dist/assets/{hooks-DjKf5pQ9.js → hooks-DuLk_j3D.js} +1 -1
  39. package/dist/assets/{html-to-image-EYrNFuGt.js → html-to-image-Nf7ijTtQ.js} +1 -1
  40. package/dist/assets/{index-DsF3qFAR.js → index-Bld_cdN5.js} +5 -5
  41. package/dist/assets/{kiosk-mode-Ht1DwVD4.js → kiosk-mode-Do8_sps4.js} +1 -1
  42. package/dist/assets/{layout-CwCyg6Oq.js → layout-BLEM1ssf.js} +3 -3
  43. package/dist/assets/{logs-panel-8V1eZbE7.js → logs-panel-5n3ctE2J.js} +1 -1
  44. package/dist/assets/{markdown-renderer-d9Lol4Mr.js → markdown-renderer-C4nMzmEI.js} +1 -1
  45. package/dist/assets/{mermaid-DAflnvW9.js → mermaid-C1kGbsPj.js} +1 -1
  46. package/dist/assets/{name-cell-input-DDR4EexG.js → name-cell-input-Cj_BDNnx.js} +1 -1
  47. package/dist/assets/{outline-panel-B-fr1jRf.js → outline-panel-xSNPEqmX.js} +1 -1
  48. package/dist/assets/{packages-panel-B4MR9Vnv.js → packages-panel-CXxRJgcn.js} +1 -1
  49. package/dist/assets/{pair-with-agent-modal-Derso4uY.js → pair-with-agent-modal-Cg38WIJM.js} +1 -1
  50. package/dist/assets/{panels-DsbaLR3B.js → panels-GDG1qxPv.js} +1 -1
  51. package/dist/assets/{process-output-CJeKhzqO.js → process-output-CExje38p.js} +1 -1
  52. package/dist/assets/{radio-group-cvI-M2aa.js → radio-group-BgIE4hAy.js} +1 -1
  53. package/dist/assets/{readonly-python-code-BmZ0KZKm.js → readonly-python-code-Cmj3Lveo.js} +1 -1
  54. package/dist/assets/renderShortcut-CUc_HCHW.js +1 -0
  55. package/dist/assets/{reveal-component-Bff44fKv.js → reveal-component-C34NH9Hx.js} +1 -1
  56. package/dist/assets/{run-page-4im7P8v3.js → run-page-NpRrbT0Y.js} +1 -1
  57. package/dist/assets/{scratchpad-panel-BFvydb89.js → scratchpad-panel-B3GWgvFG.js} +1 -1
  58. package/dist/assets/{session-panel-CwggjKWr.js → session-panel-DuKsSNdR.js} +1 -1
  59. package/dist/assets/{snippets-panel-BmWXl5GE.js → snippets-panel-BlHGRNIg.js} +1 -1
  60. package/dist/assets/{state-d_LoPKHg.js → state-Bt-mCXoz.js} +1 -1
  61. package/dist/assets/{state-G8Xj60GT.js → state-DUrsoPal.js} +3 -3
  62. package/dist/assets/{switch-YcZLs0mS.js → switch-Z7gfNPvp.js} +1 -1
  63. package/dist/assets/{terminal-BydBi1dE.js → terminal-ChwnouBc.js} +1 -1
  64. package/dist/assets/{textarea-DV6LIVoz.js → textarea-CMdRAIqd.js} +1 -1
  65. package/dist/assets/{tracing-CvLPM0Wh.js → tracing-CProSFdv.js} +1 -1
  66. package/dist/assets/{tracing-panel-iTdY5N_y.js → tracing-panel-7NGYhtTI.js} +2 -2
  67. package/dist/assets/{useBoolean-oqDq5zze.js → useBoolean-Begogf9u.js} +1 -1
  68. package/dist/assets/{useCellActionButton-CsyGJEYN.js → useCellActionButton-tt8ToMnj.js} +1 -1
  69. package/dist/assets/{useDeleteCell-BQA5MqKt.js → useDeleteCell-1tWVtBPJ.js} +1 -1
  70. package/dist/assets/{useDependencyPanelTab-CIH_2h1J.js → useDependencyPanelTab-BRZy7_Yd.js} +1 -1
  71. package/dist/assets/{useHotkey-BtqYcnxQ.js → useHotkey-CvaNHTub.js} +1 -1
  72. package/dist/assets/{useNotebookActions-jRhy52S-.js → useNotebookActions-B-lHeeZb.js} +1 -1
  73. package/dist/assets/{useRunCells-rD6HJnx6.js → useRunCells-PwtrEJj7.js} +1 -1
  74. package/dist/assets/{useSplitCell-CqznVs7m.js → useSplitCell-B86fJI0_.js} +1 -1
  75. package/dist/assets/useTheme-DJn91dnL.js +1 -0
  76. package/dist/assets/utils-Df7TLCkF.js +1 -0
  77. package/dist/assets/{vega-component-DOlcjhuV.js → vega-component-BcDGvncM.js} +1 -1
  78. package/dist/index.html +29 -29
  79. package/package.json +1 -1
  80. package/src/components/app-config/ai-config.tsx +74 -15
  81. package/src/components/chat/chat-panel.tsx +2 -2
  82. package/src/components/data-table/__tests__/header-items.test.tsx +220 -10
  83. package/src/components/data-table/column-header.tsx +17 -12
  84. package/src/components/data-table/header-items.tsx +40 -16
  85. package/src/components/editor/actions/useCellActionButton.tsx +3 -3
  86. package/src/components/editor/cell/code/cell-editor.tsx +7 -4
  87. package/src/components/editor/chrome/types.ts +13 -6
  88. package/src/components/editor/chrome/wrapper/app-chrome.tsx +6 -4
  89. package/src/components/editor/chrome/wrapper/footer-items/ai-status.tsx +10 -1
  90. package/src/components/editor/chrome/wrapper/sidebar.tsx +7 -5
  91. package/src/components/editor/errors/auto-fix.tsx +3 -3
  92. package/src/components/editor/navigation/__tests__/navigation.test.ts +15 -0
  93. package/src/components/editor/navigation/navigation.ts +5 -0
  94. package/src/components/editor/output/MarimoTracebackOutput.tsx +4 -3
  95. package/src/components/editor/renderers/cell-array.tsx +27 -24
  96. package/src/core/config/__tests__/config-schema.test.ts +2 -0
  97. package/src/core/config/config-schema.ts +1 -0
  98. package/src/core/config/config.ts +16 -0
  99. package/dist/assets/cell-editor-B-9MV21_.js +0 -20
  100. package/dist/assets/renderShortcut-CzUfgpdJ.js +0 -1
  101. package/dist/assets/useTheme-KJ1rGmnj.js +0 -1
  102. package/dist/assets/utils-Bok_hGT2.js +0 -1
@@ -0,0 +1 @@
1
+ import{d as k,i as x,l as z,p as o,u as I}from"./useEvent-D91BmmQi.js";import{A as i,B as J,I as M,J as N,N as f,P as h,R as t,T as n,U as R,b as r,w as _}from"./zod-H_cgTO0M.js";import{t as S}from"./compiler-runtime-B3qBwwSJ.js";import{t as U}from"./merge-_9WLTEyt.js";import{_ as m,c as F,g as G}from"./useEventListener-DvoEXWke.js";import{t as H}from"./invariant-BUdrueMv.js";const j=["pip","uv","rye","poetry","pixi"];var $=["normal","compact","medium","full","columns"],q=["auto","native","polars","lazy-polars","pandas"];const K="openai/gpt-4o";var O=["html","markdown","ipynb"];const L=["manual","ask","agent"];var p=h({api_key:t().optional(),base_url:t().optional(),project:t().optional()}).loose(),Q=h({chat_model:t().nullish(),edit_model:t().nullish(),autocomplete_model:t().nullish(),displayed_models:_(t()).default([]),custom_models:_(t()).default([])});const y=i({completion:h({activate_on_typing:n().prefault(!0),signature_hint_on_typing:n().prefault(!1),auto_close_pairs:n().prefault(!0),copilot:J([n(),r(["github","codeium","custom"])]).prefault(!1).transform(e=>e===!0?"github":e),codeium_api_key:t().nullish()}).prefault({}),save:i({autosave:r(["off","after_delay"]).prefault("after_delay"),autosave_delay:f().nonnegative().transform(e=>Math.max(e,1e3)).prefault(1e3),format_on_save:n().prefault(!1)}).prefault({}),formatting:i({line_length:f().nonnegative().prefault(79).transform(e=>Math.min(e,1e3))}).prefault({}),keymap:i({preset:r(["default","vim"]).prefault("default"),overrides:M(t(),t()).prefault({}),destructive_delete:n().prefault(!0)}).prefault({}),runtime:i({auto_instantiate:n().prefault(!0),on_cell_change:r(["lazy","autorun"]).prefault("autorun"),auto_reload:r(["off","lazy","autorun"]).prefault("off"),reactive_tests:n().prefault(!0),watcher_on_save:r(["lazy","autorun"]).prefault("lazy"),default_sql_output:r(q).prefault("auto"),default_auto_download:_(r(O)).prefault([]),show_tracebacks:n().prefault(!1)}).prefault({}),display:i({theme:r(["light","dark","system"]).prefault("light"),code_editor_font_size:f().nonnegative().prefault(14),cell_output:r(["above","below"]).prefault("below"),dataframes:r(["rich","plain"]).prefault("rich"),default_table_page_size:f().prefault(10),default_table_max_columns:f().prefault(50),default_width:r($).prefault("medium").transform(e=>e==="normal"?"compact":e),locale:t().nullable().optional(),reference_highlighting:n().prefault(!0)}).prefault({}),package_management:i({manager:r(j).prefault("pip")}).prefault({}),ai:i({enabled:n().prefault(!0),rules:t().prefault(""),max_tokens:f().int().positive().nullable().optional(),mode:r(L).prefault("manual"),inline_tooltip:n().prefault(!1),open_ai:p.optional(),anthropic:p.optional(),google:p.optional(),ollama:p.optional(),openrouter:p.optional(),wandb:p.optional(),opencode_go:p.optional(),open_ai_compatible:p.optional(),azure:p.optional(),bedrock:i({region_name:t().optional(),profile_name:t().optional(),aws_access_key_id:t().optional(),aws_secret_access_key:t().optional()}).optional(),custom_providers:M(t(),p).prefault({}),models:Q.prefault({displayed_models:[],custom_models:[]})}).prefault({}),experimental:i({markdown:n().optional(),rtc:n().optional()}).prefault(()=>({})),server:i({disable_file_downloads:n().optional()}).prefault(()=>({})),diagnostics:i({enabled:n().optional(),sql_linter:n().optional()}).prefault(()=>({})),sharing:i({html:n().optional(),wasm:n().optional()}).optional(),mcp:i({presets:_(r(["marimo","context7"])).optional()}).optional().prefault({})}).partial().prefault(()=>({completion:{},save:{},formatting:{},keymap:{},runtime:{},display:{},diagnostics:{},experimental:{},server:{},ai:{},package_management:{},mcp:{}})),A=t(),V=r(q).prefault("auto"),b=h({width:r($).prefault("medium").transform(e=>e==="normal"?"compact":e),app_title:A.nullish(),css_file:t().nullish(),html_head_file:t().nullish(),auto_download:_(r(O)).prefault([]),sql_output:V}).prefault(()=>({width:"medium",auto_download:[],sql_output:"auto"}));function P(e){try{return b.parse(e)}catch(a){return m.error(`Marimo got an unexpected value in the configuration file: ${a}`),b.parse({})}}function W(e){try{let a=y.parse(e);for(let[u,s]of Object.entries(a.experimental??{}))s===!0&&m.log(`\u{1F9EA} Experimental feature "${u}" is enabled.`);return a}catch(a){return a instanceof R?m.error(`Marimo got an unexpected value in the configuration file: ${N(a)}`):m.error(`Marimo got an unexpected value in the configuration file: ${a}`),T()}}function X(e){try{let a=e;return H(typeof a=="object","internal-error: marimo-config-overrides is not an object"),Object.keys(a).length>0&&m.log("\u{1F527} Project configuration overrides:",a),a}catch(a){return m.error(`Marimo got an unexpected configuration overrides: ${a}`),{}}}function T(){return y.parse({completion:{},save:{},formatting:{},keymap:{},runtime:{},display:{},diagnostics:{},experimental:{},server:{},ai:{},package_management:{},mcp:{}})}var Y=S();const v=o(T()),B=o({}),l=o(e=>{let a=e(B);return U({},e(v),a)}),Z=o(e=>e(l).runtime.auto_instantiate),ee=o(e=>e(l).keymap.overrides??{}),ae=o(G()),te=o(e=>new F(e(ee),{platform:e(ae)})),oe=o(e=>e(l).save),ne=o(e=>e(l).ai),re=o(e=>e(l).completion),le=o(e=>e(l).keymap.preset);function ie(){return z(v)}function se(){let e=(0,Y.c)(3),a=I(l),u=k(v),s;return e[0]!==a||e[1]!==u?(s=[a,u],e[0]=a,e[1]=u,e[2]=s):s=e[2],s}function ue(){return x.get(l)}const pe=o(e=>C(e(l))),fe=o(e=>D(e(l))),me=o(e=>E(e(l)));o(e=>e(l).display.code_editor_font_size);const ce=o(e=>e(l).display.locale);function C(e){var a;return((a=e.ai)==null?void 0:a.enabled)!==!1}function D(e){var a,u,s,c,d,w;return!!((u=(a=e.ai)==null?void 0:a.models)!=null&&u.chat_model)||!!((c=(s=e.ai)==null?void 0:s.models)!=null&&c.edit_model)||!!((w=(d=e.ai)==null?void 0:d.models)!=null&&w.autocomplete_model)}function E(e){return C(e)&&D(e)}const g=o(P({}));function de(){return z(g)}function _e(){return k(g)}function ge(){return x.get(g)}const he=o(e=>e(g).width);o(e=>{var c,d;let a=e(l),u=((c=a.snippets)==null?void 0:c.custom_paths)??[],s=(d=a.snippets)==null?void 0:d.include_default_snippets;return u.length>0||s===!0});const ve=o(e=>{var a;return((a=e(l).server)==null?void 0:a.disable_file_downloads)??!1});function ye(){return!1}export{P as A,ie as C,K as D,A as E,W as M,j as O,_e as S,b as T,le as _,fe as a,de as b,Z as c,B as d,ve as f,E as g,te as h,me as i,X as j,y as k,oe as l,ue as m,ne as n,g as o,ge as p,pe as r,he as s,ye as t,re as u,ce as v,v as w,se as x,l as y};
@@ -1 +1 @@
1
- import{s as Z}from"./chunk-LvLJmgfZ.js";import{n as C}from"./useEvent-D91BmmQi.js";import{t as Y}from"./react-Bj1aDYRI.js";import"./react-dom-CSu739Rf.js";import{t as tt}from"./compiler-runtime-B3qBwwSJ.js";import{t as et}from"./debounce-DhnxH9Rh.js";import{_,d as V}from"./useEventListener-DvoEXWke.js";import{n as H}from"./config-hed0CaQL.js";import{t as nt}from"./cn-DCUzRj2J.js";import{t as rt}from"./jsx-runtime-BqBOg78p.js";import"./fullscreen-eipL3i3Y.js";import{t as at}from"./tooltip-DTV9tlSr.js";import{r as it}from"./button-BbCh-29a.js";import{d as N}from"./arrays-DYDL-3-i.js";import{u as ot}from"./toDate-B5A0DFEz.js";import"./session-0B5NBztP.js";import{r as lt}from"./useTheme-KJ1rGmnj.js";import{t as ct}from"./isValid-CklTTytn.js";import{a as st,r as mt,t as pt}from"./utils-DCL4n9wx.js";import{n as ut}from"./vega-loader.browser-C8wT63Va.js";import{t as ft}from"./react-vega-CXIQBUis.js";import"./defaultLocale-BLUna9fQ.js";import"./defaultLocale-DzliDDTm.js";import{r as dt,t as ht}from"./alert-yTS3WpF4.js";import{n as gt}from"./error-banner-LdWZDbqd.js";import{t as yt}from"./formats-b7Sf6DAK.js";import{n as vt}from"./useAsyncData-bgszE9F0.js";import{t as F}from"./useDeepCompareMemoize-zUHU--0D.js";import{t as kt}from"./semaphore-FlZezxaf.js";var bt=tt(),S=Z(Y(),1);function wt(t){return t.data&&"url"in t.data&&(t.data.url=H(t.data.url).href),t}const u={arc:"arc",area:"area",bar:"bar",image:"image",line:"line",point:"point",rect:"rect",rule:"rule",text:"text",tick:"tick",trail:"trail",circle:"circle",square:"square",geoshape:"geoshape"};var W=new Set(["boxplot","errorband","errorbar"]);const x={getMarkType(t){let e=typeof t=="string"?t:t.type;if(W.has(e))throw Error("Not supported");return e},isInteractive(t){let e=typeof t=="string"?t:t.type;return!W.has(e)},makeClickable(t){let e=typeof t=="string"?t:t.type;return e in u?typeof t=="string"?{type:t,cursor:"pointer",tooltip:!0}:{...t,type:e,cursor:"pointer",tooltip:!0}:t},getOpacity(t){return typeof t=="string"?null:"opacity"in t&&typeof t.opacity=="number"?t.opacity:null}},v={point(t){return t==null?"select_point":`select_point_${t}`},interval(t){return t==null?"select_interval":`select_interval_${t}`},legendSelection(t){return`legend_selection_${t}`},binColoring(t){return t==null?"bin_coloring":`bin_coloring_${t}`},HIGHLIGHT:"highlight",PAN_ZOOM:"pan_zoom",hasPoint(t){return t.some(e=>e.startsWith("select_point"))},hasInterval(t){return t.some(e=>e.startsWith("select_interval"))},hasLegend(t){return t.some(e=>e.startsWith("legend_selection"))},hasPanZoom(t){return t.some(e=>e.startsWith("pan_zoom"))},isBinColoring(t){return t.startsWith("bin_coloring")}},j={highlight(){return{name:v.HIGHLIGHT,select:{type:"point",on:"mouseover"}}},interval(t,e){return{name:v.interval(e),select:{type:"interval",encodings:D(t),mark:{fill:"#669EFF",fillOpacity:.07,stroke:"#669EFF",strokeOpacity:.4},on:"[mousedown[!event.metaKey], mouseup] > mousemove[!event.metaKey]",translate:"[mousedown[!event.metaKey], mouseup] > mousemove[!event.metaKey]"}}},point(t,e){return{name:v.point(e),select:{type:"point",encodings:D(t),on:"click[!event.metaKey]"}}},binColoring(t){return{name:v.binColoring(t),select:{type:"point",on:"click[!event.metaKey]"}}},legend(t){return{name:v.legendSelection(t),select:{type:"point",fields:[t]},bind:"legend"}},panZoom(){return{name:v.PAN_ZOOM,bind:"scales",select:{type:"interval",on:"[mousedown[event.metaKey], window:mouseup] > window:mousemove!",translate:"[mousedown[event.metaKey], window:mouseup] > window:mousemove!",zoom:"wheel![event.metaKey]"}}}};function D(t){switch(x.getMarkType(t.mark)){case u.image:case u.trail:return;case u.area:case u.arc:return["color"];case u.bar:{let e=St(t);return e==="horizontal"?["y"]:e==="vertical"?["x"]:void 0}case u.circle:case u.geoshape:case u.line:case u.point:case u.rect:case u.rule:case u.square:case u.text:case u.tick:return["x","y"]}}function P(t){return"params"in t&&t.params&&t.params.length>0?N(t.params.filter(e=>e==null?!1:"select"in e&&e.select!==void 0).map(e=>e.name)):"layer"in t?N(t.layer.flatMap(P)):"vconcat"in t?N(t.vconcat.flatMap(P)):"hconcat"in t?N(t.hconcat.flatMap(P)):[]}function St(t){var a,i;if(!t||!("mark"in t))return;let e=(a=t.encoding)==null?void 0:a.x,n=(i=t.encoding)==null?void 0:i.y;if(e&&"type"in e&&e.type==="nominal")return"vertical";if(n&&"type"in n&&n.type==="nominal"||e&&"aggregate"in e)return"horizontal";if(n&&"aggregate"in n)return"vertical"}function xt(t){if(!t.encoding)return[];let e=[];for(let n of Object.values(t.encoding))n&&typeof n=="object"&&"bin"in n&&n.bin&&"field"in n&&typeof n.field=="string"&&e.push(n.field);return e}function G(t){if(!t||!("encoding"in t))return[];let{encoding:e}=t;return e?Object.entries(e).flatMap(n=>{let[a,i]=n;return!i||!At.has(a)?[]:"field"in i&&typeof i.field=="string"?[i.field]:"condition"in i&&i.condition&&typeof i.condition=="object"&&"field"in i.condition&&i.condition.field&&typeof i.condition.field=="string"?[i.condition.field]:[]}):[]}var At=new Set(["color","fill","fillOpacity","opacity","shape","size"]);function $(t,e,n,a){let i=n.filter(o=>v.isBinColoring(o)),r={and:(i.length>0?i:n).map(o=>({param:o}))};if(t==="opacity"){let o=x.getOpacity(a)||1;return{...e,opacity:{condition:{test:r,value:o},value:o/5}}}else return e}function _t(t){if(!("select"in t)||!t.select)return JSON.stringify(t);let e=t.select;if(typeof e=="string")return JSON.stringify({type:e,bind:t.bind});let n={type:e.type,encodings:"encodings"in e&&e.encodings?[...e.encodings].toSorted():void 0,fields:"fields"in e&&e.fields?[...e.fields].toSorted():void 0,bind:t.bind};return JSON.stringify(n)}function q(t){let e=E(t);if(e.length===0)return t;let n=jt(e);return n.length===0?t:{...L(K(t,new Set(n.map(a=>a.name))),n.map(a=>a.name)),params:[...t.params||[],...n]}}function E(t){let e=[];if("vconcat"in t&&Array.isArray(t.vconcat))for(let n of t.vconcat)e.push(...E(n));else if("hconcat"in t&&Array.isArray(t.hconcat))for(let n of t.hconcat)e.push(...E(n));else{if("layer"in t)return[];"mark"in t&&"params"in t&&t.params&&t.params.length>0&&e.push({params:t.params})}return e}function jt(t){if(t.length===0)return[];let e=new Map,n=t.length;for(let{params:i}of t){let r=new Set;for(let o of i){let s=_t(o);r.has(s)||(r.add(s),e.has(s)||e.set(s,{count:0,param:o}),e.get(s).count++)}}let a=[];for(let[,{count:i,param:r}]of e)i===n&&a.push(r);return a}function K(t,e){if("vconcat"in t&&Array.isArray(t.vconcat))return{...t,vconcat:t.vconcat.map(n=>K(n,e))};if("hconcat"in t&&Array.isArray(t.hconcat))return{...t,hconcat:t.hconcat.map(n=>K(n,e))};if("mark"in t&&"params"in t&&t.params){let n=t.params,a=[];for(let i of n){if(!i||typeof i!="object"||!("name"in i)){a.push(i);continue}e.has(i.name)||a.push(i)}if(a.length===0){let{params:i,...r}=t;return r}return{...t,params:a}}return t}function L(t,e){return"vconcat"in t&&Array.isArray(t.vconcat)?{...t,vconcat:t.vconcat.map(n=>L(n,e))}:"hconcat"in t&&Array.isArray(t.hconcat)?{...t,hconcat:t.hconcat.map(n=>L(n,e))}:"layer"in t?t:"mark"in t&&x.isInteractive(t.mark)?{...t,mark:x.makeClickable(t.mark),encoding:$("opacity",t.encoding||{},e,t.mark)}:t}function T(t,e){var s,h;let{chartSelection:n=!0,fieldSelection:a=!0}=e;if(!n&&!a)return t;(s=t.params)!=null&&s.some(l=>l.bind==="legend")&&(a=!1);let i=(h=t.params)==null?void 0:h.some(l=>!l.bind);i&&(n=!1);let r="vconcat"in t||"hconcat"in t;if(i&&r)return t;if("vconcat"in t){let l=t.vconcat.map(m=>"mark"in m?T(m,{chartSelection:n,fieldSelection:a}):m);return q({...t,vconcat:l})}if("hconcat"in t){let l=t.hconcat.map(m=>"mark"in m?T(m,{chartSelection:n,fieldSelection:a}):m);return q({...t,hconcat:l})}if("layer"in t){let l=t.params&&t.params.length>0,m=a!==!1&&!l,k=[];if(m){let p=t.layer.flatMap(f=>"mark"in f?G(f):[]);k=[...new Set(p)],Array.isArray(a)&&(k=k.filter(f=>a.includes(f)))}let w=t.layer.map((p,f)=>{if(!("mark"in p))return p;let g=p;if(f===0&&k.length>0){let O=k.map(M=>j.legend(M));g={...g,params:[...g.params||[],...O]}}return g=z(g,n,f),g=J(g),f===0&&(g=B(g)),g});return{...t,layer:w}}if(!("mark"in t)||!x.isInteractive(t.mark))return t;let o=t;return o=Ot(o,a),o=z(o,n,void 0),o=J(o),o=B(o),o}function Ot(t,e){if(e===!1)return t;let n=G(t);Array.isArray(e)&&(n=n.filter(r=>e.includes(r)));let a=n.map(r=>j.legend(r)),i=[...t.params||[],...a];return{...t,params:i}}function z(t,e,n){if(e===!1)return t;let a;try{a=x.getMarkType(t.mark)}catch{return t}if(a==="geoshape")return t;let i=xt(t),r=e===!0?i.length>0?["point"]:Mt(a):[e];if(!r||r.length===0)return t;let o=r.map(h=>h==="interval"?j.interval(t,n):j.point(t,n)),s=[...t.params||[],...o];return i.length>0&&r.includes("point")&&s.push(j.binColoring(n)),{...t,params:s}}function B(t){let e;try{e=x.getMarkType(t.mark)}catch{}if(e==="geoshape")return t;let n=t.params||[];return n.some(a=>a.bind==="scales")?t:{...t,params:[...n,j.panZoom()]}}function J(t){let e="encoding"in t?t.encoding:void 0,n=t.params||[],a=n.map(i=>i.name);return n.length===0||!x.isInteractive(t.mark)?t:{...t,mark:x.makeClickable(t.mark),encoding:$("opacity",e||{},a,t.mark)}}function Mt(t){switch(t){case"arc":case"area":return["point"];case"text":case"bar":return["point","interval"];case"line":return;default:return["point","interval"]}}var Nt=5;async function Pt(t){if(!t)return t;let e="datasets"in t?{...t.datasets}:{},n=new kt(Nt),a=async r=>{if(!r)return r;if("layer"in r){let l=await Promise.all(r.layer.map(a));r={...r,layer:l}}if("hconcat"in r){let l=await Promise.all(r.hconcat.map(a));r={...r,hconcat:l}}if("vconcat"in r){let l=await Promise.all(r.vconcat.map(a));r={...r,vconcat:l}}if("spec"in r&&(r={...r,spec:await a(r.spec)}),!r.data||!("url"in r.data))return r;let o;try{o=H(r.data.url)}catch{return r}let s=r.data.format,h=await n.run(()=>st(o.href,s));return e[o.pathname]=h,{...r,data:{name:o.pathname}}},i=await a(t);return Object.keys(e).length===0?i:{...i,datasets:e}}var d=Z(rt(),1);ut("arrow",yt);var It=t=>{let e=(0,bt.c)(12),{value:n,setValue:a,chartSelection:i,fieldSelection:r,spec:o,embedOptions:s}=t,h,l;e[0]===o?(h=e[1],l=e[2]):(h=async()=>Pt(o),l=[o],e[0]=o,e[1]=h,e[2]=l);let{data:m,error:k}=vt(h,l);if(k){let p;return e[3]===k?p=e[4]:(p=(0,d.jsx)(gt,{error:k}),e[3]=k,e[4]=p),p}if(!m)return null;let w;return e[5]!==i||e[6]!==s||e[7]!==r||e[8]!==m||e[9]!==a||e[10]!==n?(w=(0,d.jsx)(Ct,{value:n,setValue:a,chartSelection:i,fieldSelection:r,spec:m,embedOptions:s}),e[5]=i,e[6]=s,e[7]=r,e[8]=m,e[9]=a,e[10]=n,e[11]=w):w=e[11],w},Ct=({value:t,setValue:e,chartSelection:n,fieldSelection:a,spec:i,embedOptions:r})=>{let{theme:o}=lt(),s=(0,S.useRef)(null),h=(0,S.useRef)(void 0),[l,m]=(0,S.useState)(),k=(0,S.useMemo)(()=>r&&"actions"in r?r.actions:{source:!1,compiled:!1},[r]),w=F(i),p=(0,S.useMemo)(()=>T(wt(w),{chartSelection:n,fieldSelection:a}),[w,n,a]),f=(0,S.useMemo)(()=>P(p),[p]),g=C(c=>{e({...t,...c})}),O=(0,S.useMemo)(()=>et((c,y)=>{_.debug("[Vega signal]",c,y);let b=V.mapValues(y,Kt);b=V.mapValues(b,Et),g({[c]:b})},100),[g]),M=F(f),I=(0,S.useMemo)(()=>M.reduce((c,y)=>(v.PAN_ZOOM===y||v.isBinColoring(y)||c.push({signalName:y,handler:(b,X)=>O(b,X)}),c),[]),[M,O]),R=C(c=>{_.error(c),_.debug(p),m(c)}),Q=C(c=>{_.debug("[Vega view] created",c),h.current=c,m(void 0)}),U=()=>{let c=[];return v.hasPoint(f)&&c.push(["Point selection","click to select a point; hold shift for multi-select"]),v.hasInterval(f)&&c.push(["Interval selection","click and drag to select an interval"]),v.hasLegend(f)&&c.push(["Legend selection","click to select a legend item; hold shift for multi-select"]),v.hasPanZoom(f)&&c.push(["Pan","hold the meta key and drag"],["Zoom","hold the meta key and scroll"]),c.length===0?null:(0,d.jsx)(at,{delayDuration:300,side:"left",content:(0,d.jsx)("div",{className:"text-xs flex flex-col",children:c.map((y,b)=>(0,d.jsxs)("div",{children:[(0,d.jsxs)("span",{className:"font-bold tracking-wide",children:[y[0],":"]})," ",y[1]]},b))}),children:(0,d.jsx)(ot,{className:"absolute bottom-1 right-0 m-2 h-4 w-4 cursor-help text-muted-foreground hover:text-foreground"})})},A=ft({ref:s,spec:p,options:{theme:o==="dark"?"dark":void 0,actions:k,mode:"vega-lite",tooltip:mt.call,renderer:"canvas"},onError:R,onEmbed:Q});return(0,S.useEffect)(()=>(I.forEach(({signalName:c,handler:y})=>{try{A==null||A.view.addSignalListener(c,y)}catch(b){_.error(b)}}),()=>{I.forEach(({signalName:c,handler:y})=>{try{A==null||A.view.removeSignalListener(c,y)}catch(b){_.error(b)}})}),[A,I]),(0,d.jsxs)(d.Fragment,{children:[l&&(0,d.jsxs)(ht,{variant:"destructive",children:[(0,d.jsx)(dt,{children:l.message}),(0,d.jsx)("div",{className:"text-md",children:l.stack})]}),(0,d.jsxs)("div",{className:nt("relative"),onPointerDown:it.stopPropagation(),children:[(0,d.jsx)("div",{ref:s,"data-container-width":pt(p)}),U()]})]})};function Et(t){return t instanceof Set?[...t]:t}function Kt(t){return Array.isArray(t)?t.map(e=>e instanceof Date&&ct(e)?new Date(e).getTime():e):t}var Lt=It;export{Lt as default};
1
+ import{s as Z}from"./chunk-LvLJmgfZ.js";import{n as C}from"./useEvent-D91BmmQi.js";import{t as Y}from"./react-Bj1aDYRI.js";import"./react-dom-CSu739Rf.js";import{t as tt}from"./compiler-runtime-B3qBwwSJ.js";import{t as et}from"./debounce-DhnxH9Rh.js";import{_,d as V}from"./useEventListener-DvoEXWke.js";import{n as H}from"./config-BG-ACF7_.js";import{t as nt}from"./cn-DCUzRj2J.js";import{t as rt}from"./jsx-runtime-BqBOg78p.js";import"./fullscreen-eipL3i3Y.js";import{t as at}from"./tooltip-DTV9tlSr.js";import{r as it}from"./button-BbCh-29a.js";import{d as N}from"./arrays-DYDL-3-i.js";import{u as ot}from"./toDate-B5A0DFEz.js";import"./session-0B5NBztP.js";import{r as lt}from"./useTheme-DJn91dnL.js";import{t as ct}from"./isValid-CklTTytn.js";import{a as st,r as mt,t as pt}from"./utils-DCL4n9wx.js";import{n as ut}from"./vega-loader.browser-C8wT63Va.js";import{t as ft}from"./react-vega-CXIQBUis.js";import"./defaultLocale-BLUna9fQ.js";import"./defaultLocale-DzliDDTm.js";import{r as dt,t as ht}from"./alert-yTS3WpF4.js";import{n as gt}from"./error-banner-LdWZDbqd.js";import{t as yt}from"./formats-b7Sf6DAK.js";import{n as vt}from"./useAsyncData-bgszE9F0.js";import{t as F}from"./useDeepCompareMemoize-zUHU--0D.js";import{t as kt}from"./semaphore-FlZezxaf.js";var bt=tt(),S=Z(Y(),1);function wt(t){return t.data&&"url"in t.data&&(t.data.url=H(t.data.url).href),t}const u={arc:"arc",area:"area",bar:"bar",image:"image",line:"line",point:"point",rect:"rect",rule:"rule",text:"text",tick:"tick",trail:"trail",circle:"circle",square:"square",geoshape:"geoshape"};var W=new Set(["boxplot","errorband","errorbar"]);const x={getMarkType(t){let e=typeof t=="string"?t:t.type;if(W.has(e))throw Error("Not supported");return e},isInteractive(t){let e=typeof t=="string"?t:t.type;return!W.has(e)},makeClickable(t){let e=typeof t=="string"?t:t.type;return e in u?typeof t=="string"?{type:t,cursor:"pointer",tooltip:!0}:{...t,type:e,cursor:"pointer",tooltip:!0}:t},getOpacity(t){return typeof t=="string"?null:"opacity"in t&&typeof t.opacity=="number"?t.opacity:null}},v={point(t){return t==null?"select_point":`select_point_${t}`},interval(t){return t==null?"select_interval":`select_interval_${t}`},legendSelection(t){return`legend_selection_${t}`},binColoring(t){return t==null?"bin_coloring":`bin_coloring_${t}`},HIGHLIGHT:"highlight",PAN_ZOOM:"pan_zoom",hasPoint(t){return t.some(e=>e.startsWith("select_point"))},hasInterval(t){return t.some(e=>e.startsWith("select_interval"))},hasLegend(t){return t.some(e=>e.startsWith("legend_selection"))},hasPanZoom(t){return t.some(e=>e.startsWith("pan_zoom"))},isBinColoring(t){return t.startsWith("bin_coloring")}},j={highlight(){return{name:v.HIGHLIGHT,select:{type:"point",on:"mouseover"}}},interval(t,e){return{name:v.interval(e),select:{type:"interval",encodings:D(t),mark:{fill:"#669EFF",fillOpacity:.07,stroke:"#669EFF",strokeOpacity:.4},on:"[mousedown[!event.metaKey], mouseup] > mousemove[!event.metaKey]",translate:"[mousedown[!event.metaKey], mouseup] > mousemove[!event.metaKey]"}}},point(t,e){return{name:v.point(e),select:{type:"point",encodings:D(t),on:"click[!event.metaKey]"}}},binColoring(t){return{name:v.binColoring(t),select:{type:"point",on:"click[!event.metaKey]"}}},legend(t){return{name:v.legendSelection(t),select:{type:"point",fields:[t]},bind:"legend"}},panZoom(){return{name:v.PAN_ZOOM,bind:"scales",select:{type:"interval",on:"[mousedown[event.metaKey], window:mouseup] > window:mousemove!",translate:"[mousedown[event.metaKey], window:mouseup] > window:mousemove!",zoom:"wheel![event.metaKey]"}}}};function D(t){switch(x.getMarkType(t.mark)){case u.image:case u.trail:return;case u.area:case u.arc:return["color"];case u.bar:{let e=St(t);return e==="horizontal"?["y"]:e==="vertical"?["x"]:void 0}case u.circle:case u.geoshape:case u.line:case u.point:case u.rect:case u.rule:case u.square:case u.text:case u.tick:return["x","y"]}}function P(t){return"params"in t&&t.params&&t.params.length>0?N(t.params.filter(e=>e==null?!1:"select"in e&&e.select!==void 0).map(e=>e.name)):"layer"in t?N(t.layer.flatMap(P)):"vconcat"in t?N(t.vconcat.flatMap(P)):"hconcat"in t?N(t.hconcat.flatMap(P)):[]}function St(t){var a,i;if(!t||!("mark"in t))return;let e=(a=t.encoding)==null?void 0:a.x,n=(i=t.encoding)==null?void 0:i.y;if(e&&"type"in e&&e.type==="nominal")return"vertical";if(n&&"type"in n&&n.type==="nominal"||e&&"aggregate"in e)return"horizontal";if(n&&"aggregate"in n)return"vertical"}function xt(t){if(!t.encoding)return[];let e=[];for(let n of Object.values(t.encoding))n&&typeof n=="object"&&"bin"in n&&n.bin&&"field"in n&&typeof n.field=="string"&&e.push(n.field);return e}function G(t){if(!t||!("encoding"in t))return[];let{encoding:e}=t;return e?Object.entries(e).flatMap(n=>{let[a,i]=n;return!i||!At.has(a)?[]:"field"in i&&typeof i.field=="string"?[i.field]:"condition"in i&&i.condition&&typeof i.condition=="object"&&"field"in i.condition&&i.condition.field&&typeof i.condition.field=="string"?[i.condition.field]:[]}):[]}var At=new Set(["color","fill","fillOpacity","opacity","shape","size"]);function $(t,e,n,a){let i=n.filter(o=>v.isBinColoring(o)),r={and:(i.length>0?i:n).map(o=>({param:o}))};if(t==="opacity"){let o=x.getOpacity(a)||1;return{...e,opacity:{condition:{test:r,value:o},value:o/5}}}else return e}function _t(t){if(!("select"in t)||!t.select)return JSON.stringify(t);let e=t.select;if(typeof e=="string")return JSON.stringify({type:e,bind:t.bind});let n={type:e.type,encodings:"encodings"in e&&e.encodings?[...e.encodings].toSorted():void 0,fields:"fields"in e&&e.fields?[...e.fields].toSorted():void 0,bind:t.bind};return JSON.stringify(n)}function q(t){let e=E(t);if(e.length===0)return t;let n=jt(e);return n.length===0?t:{...L(K(t,new Set(n.map(a=>a.name))),n.map(a=>a.name)),params:[...t.params||[],...n]}}function E(t){let e=[];if("vconcat"in t&&Array.isArray(t.vconcat))for(let n of t.vconcat)e.push(...E(n));else if("hconcat"in t&&Array.isArray(t.hconcat))for(let n of t.hconcat)e.push(...E(n));else{if("layer"in t)return[];"mark"in t&&"params"in t&&t.params&&t.params.length>0&&e.push({params:t.params})}return e}function jt(t){if(t.length===0)return[];let e=new Map,n=t.length;for(let{params:i}of t){let r=new Set;for(let o of i){let s=_t(o);r.has(s)||(r.add(s),e.has(s)||e.set(s,{count:0,param:o}),e.get(s).count++)}}let a=[];for(let[,{count:i,param:r}]of e)i===n&&a.push(r);return a}function K(t,e){if("vconcat"in t&&Array.isArray(t.vconcat))return{...t,vconcat:t.vconcat.map(n=>K(n,e))};if("hconcat"in t&&Array.isArray(t.hconcat))return{...t,hconcat:t.hconcat.map(n=>K(n,e))};if("mark"in t&&"params"in t&&t.params){let n=t.params,a=[];for(let i of n){if(!i||typeof i!="object"||!("name"in i)){a.push(i);continue}e.has(i.name)||a.push(i)}if(a.length===0){let{params:i,...r}=t;return r}return{...t,params:a}}return t}function L(t,e){return"vconcat"in t&&Array.isArray(t.vconcat)?{...t,vconcat:t.vconcat.map(n=>L(n,e))}:"hconcat"in t&&Array.isArray(t.hconcat)?{...t,hconcat:t.hconcat.map(n=>L(n,e))}:"layer"in t?t:"mark"in t&&x.isInteractive(t.mark)?{...t,mark:x.makeClickable(t.mark),encoding:$("opacity",t.encoding||{},e,t.mark)}:t}function T(t,e){var s,h;let{chartSelection:n=!0,fieldSelection:a=!0}=e;if(!n&&!a)return t;(s=t.params)!=null&&s.some(l=>l.bind==="legend")&&(a=!1);let i=(h=t.params)==null?void 0:h.some(l=>!l.bind);i&&(n=!1);let r="vconcat"in t||"hconcat"in t;if(i&&r)return t;if("vconcat"in t){let l=t.vconcat.map(m=>"mark"in m?T(m,{chartSelection:n,fieldSelection:a}):m);return q({...t,vconcat:l})}if("hconcat"in t){let l=t.hconcat.map(m=>"mark"in m?T(m,{chartSelection:n,fieldSelection:a}):m);return q({...t,hconcat:l})}if("layer"in t){let l=t.params&&t.params.length>0,m=a!==!1&&!l,k=[];if(m){let p=t.layer.flatMap(f=>"mark"in f?G(f):[]);k=[...new Set(p)],Array.isArray(a)&&(k=k.filter(f=>a.includes(f)))}let w=t.layer.map((p,f)=>{if(!("mark"in p))return p;let g=p;if(f===0&&k.length>0){let O=k.map(M=>j.legend(M));g={...g,params:[...g.params||[],...O]}}return g=z(g,n,f),g=J(g),f===0&&(g=B(g)),g});return{...t,layer:w}}if(!("mark"in t)||!x.isInteractive(t.mark))return t;let o=t;return o=Ot(o,a),o=z(o,n,void 0),o=J(o),o=B(o),o}function Ot(t,e){if(e===!1)return t;let n=G(t);Array.isArray(e)&&(n=n.filter(r=>e.includes(r)));let a=n.map(r=>j.legend(r)),i=[...t.params||[],...a];return{...t,params:i}}function z(t,e,n){if(e===!1)return t;let a;try{a=x.getMarkType(t.mark)}catch{return t}if(a==="geoshape")return t;let i=xt(t),r=e===!0?i.length>0?["point"]:Mt(a):[e];if(!r||r.length===0)return t;let o=r.map(h=>h==="interval"?j.interval(t,n):j.point(t,n)),s=[...t.params||[],...o];return i.length>0&&r.includes("point")&&s.push(j.binColoring(n)),{...t,params:s}}function B(t){let e;try{e=x.getMarkType(t.mark)}catch{}if(e==="geoshape")return t;let n=t.params||[];return n.some(a=>a.bind==="scales")?t:{...t,params:[...n,j.panZoom()]}}function J(t){let e="encoding"in t?t.encoding:void 0,n=t.params||[],a=n.map(i=>i.name);return n.length===0||!x.isInteractive(t.mark)?t:{...t,mark:x.makeClickable(t.mark),encoding:$("opacity",e||{},a,t.mark)}}function Mt(t){switch(t){case"arc":case"area":return["point"];case"text":case"bar":return["point","interval"];case"line":return;default:return["point","interval"]}}var Nt=5;async function Pt(t){if(!t)return t;let e="datasets"in t?{...t.datasets}:{},n=new kt(Nt),a=async r=>{if(!r)return r;if("layer"in r){let l=await Promise.all(r.layer.map(a));r={...r,layer:l}}if("hconcat"in r){let l=await Promise.all(r.hconcat.map(a));r={...r,hconcat:l}}if("vconcat"in r){let l=await Promise.all(r.vconcat.map(a));r={...r,vconcat:l}}if("spec"in r&&(r={...r,spec:await a(r.spec)}),!r.data||!("url"in r.data))return r;let o;try{o=H(r.data.url)}catch{return r}let s=r.data.format,h=await n.run(()=>st(o.href,s));return e[o.pathname]=h,{...r,data:{name:o.pathname}}},i=await a(t);return Object.keys(e).length===0?i:{...i,datasets:e}}var d=Z(rt(),1);ut("arrow",yt);var It=t=>{let e=(0,bt.c)(12),{value:n,setValue:a,chartSelection:i,fieldSelection:r,spec:o,embedOptions:s}=t,h,l;e[0]===o?(h=e[1],l=e[2]):(h=async()=>Pt(o),l=[o],e[0]=o,e[1]=h,e[2]=l);let{data:m,error:k}=vt(h,l);if(k){let p;return e[3]===k?p=e[4]:(p=(0,d.jsx)(gt,{error:k}),e[3]=k,e[4]=p),p}if(!m)return null;let w;return e[5]!==i||e[6]!==s||e[7]!==r||e[8]!==m||e[9]!==a||e[10]!==n?(w=(0,d.jsx)(Ct,{value:n,setValue:a,chartSelection:i,fieldSelection:r,spec:m,embedOptions:s}),e[5]=i,e[6]=s,e[7]=r,e[8]=m,e[9]=a,e[10]=n,e[11]=w):w=e[11],w},Ct=({value:t,setValue:e,chartSelection:n,fieldSelection:a,spec:i,embedOptions:r})=>{let{theme:o}=lt(),s=(0,S.useRef)(null),h=(0,S.useRef)(void 0),[l,m]=(0,S.useState)(),k=(0,S.useMemo)(()=>r&&"actions"in r?r.actions:{source:!1,compiled:!1},[r]),w=F(i),p=(0,S.useMemo)(()=>T(wt(w),{chartSelection:n,fieldSelection:a}),[w,n,a]),f=(0,S.useMemo)(()=>P(p),[p]),g=C(c=>{e({...t,...c})}),O=(0,S.useMemo)(()=>et((c,y)=>{_.debug("[Vega signal]",c,y);let b=V.mapValues(y,Kt);b=V.mapValues(b,Et),g({[c]:b})},100),[g]),M=F(f),I=(0,S.useMemo)(()=>M.reduce((c,y)=>(v.PAN_ZOOM===y||v.isBinColoring(y)||c.push({signalName:y,handler:(b,X)=>O(b,X)}),c),[]),[M,O]),R=C(c=>{_.error(c),_.debug(p),m(c)}),Q=C(c=>{_.debug("[Vega view] created",c),h.current=c,m(void 0)}),U=()=>{let c=[];return v.hasPoint(f)&&c.push(["Point selection","click to select a point; hold shift for multi-select"]),v.hasInterval(f)&&c.push(["Interval selection","click and drag to select an interval"]),v.hasLegend(f)&&c.push(["Legend selection","click to select a legend item; hold shift for multi-select"]),v.hasPanZoom(f)&&c.push(["Pan","hold the meta key and drag"],["Zoom","hold the meta key and scroll"]),c.length===0?null:(0,d.jsx)(at,{delayDuration:300,side:"left",content:(0,d.jsx)("div",{className:"text-xs flex flex-col",children:c.map((y,b)=>(0,d.jsxs)("div",{children:[(0,d.jsxs)("span",{className:"font-bold tracking-wide",children:[y[0],":"]})," ",y[1]]},b))}),children:(0,d.jsx)(ot,{className:"absolute bottom-1 right-0 m-2 h-4 w-4 cursor-help text-muted-foreground hover:text-foreground"})})},A=ft({ref:s,spec:p,options:{theme:o==="dark"?"dark":void 0,actions:k,mode:"vega-lite",tooltip:mt.call,renderer:"canvas"},onError:R,onEmbed:Q});return(0,S.useEffect)(()=>(I.forEach(({signalName:c,handler:y})=>{try{A==null||A.view.addSignalListener(c,y)}catch(b){_.error(b)}}),()=>{I.forEach(({signalName:c,handler:y})=>{try{A==null||A.view.removeSignalListener(c,y)}catch(b){_.error(b)}})}),[A,I]),(0,d.jsxs)(d.Fragment,{children:[l&&(0,d.jsxs)(ht,{variant:"destructive",children:[(0,d.jsx)(dt,{children:l.message}),(0,d.jsx)("div",{className:"text-md",children:l.stack})]}),(0,d.jsxs)("div",{className:nt("relative"),onPointerDown:it.stopPropagation(),children:[(0,d.jsx)("div",{ref:s,"data-container-width":pt(p)}),U()]})]})};function Et(t){return t instanceof Set?[...t]:t}function Kt(t){return Array.isArray(t)?t.map(e=>e instanceof Date&&ct(e)?new Date(e).getTime():e):t}var Lt=It;export{Lt as default};
package/dist/index.html CHANGED
@@ -66,7 +66,7 @@
66
66
  <marimo-server-token data-token="{{ server_token }}" hidden></marimo-server-token>
67
67
  <!-- /TODO -->
68
68
  <title>{{ title }}</title>
69
- <script type="module" crossorigin src="./assets/index-DsF3qFAR.js"></script>
69
+ <script type="module" crossorigin src="./assets/index-Bld_cdN5.js"></script>
70
70
  <link rel="modulepreload" crossorigin href="./assets/preload-helper-BPPi7vOr.js">
71
71
  <link rel="modulepreload" crossorigin href="./assets/chunk-LvLJmgfZ.js">
72
72
  <link rel="modulepreload" crossorigin href="./assets/react-Bj1aDYRI.js">
@@ -98,17 +98,17 @@
98
98
  <link rel="modulepreload" crossorigin href="./assets/_baseFor-DKD1r8uL.js">
99
99
  <link rel="modulepreload" crossorigin href="./assets/merge-_9WLTEyt.js">
100
100
  <link rel="modulepreload" crossorigin href="./assets/zod-H_cgTO0M.js">
101
- <link rel="modulepreload" crossorigin href="./assets/utils-Bok_hGT2.js">
101
+ <link rel="modulepreload" crossorigin href="./assets/utils-Df7TLCkF.js">
102
102
  <link rel="modulepreload" crossorigin href="./assets/Deferred-CP7vq682.js">
103
103
  <link rel="modulepreload" crossorigin href="./assets/uuid-e9lSomwA.js">
104
104
  <link rel="modulepreload" crossorigin href="./assets/DeferredRequestRegistry-CNldVcrP.js">
105
105
  <link rel="modulepreload" crossorigin href="./assets/constants-DMpttj8Q.js">
106
106
  <link rel="modulepreload" crossorigin href="./assets/session-0B5NBztP.js">
107
- <link rel="modulepreload" crossorigin href="./assets/config-hed0CaQL.js">
107
+ <link rel="modulepreload" crossorigin href="./assets/config-BG-ACF7_.js">
108
108
  <link rel="modulepreload" crossorigin href="./assets/requests-DIwGYs0l.js">
109
109
  <link rel="modulepreload" crossorigin href="./assets/useLifecycle-DieWOfXE.js">
110
110
  <link rel="modulepreload" crossorigin href="./assets/useNonce-DfoVjkkH.js">
111
- <link rel="modulepreload" crossorigin href="./assets/useTheme-KJ1rGmnj.js">
111
+ <link rel="modulepreload" crossorigin href="./assets/useTheme-DJn91dnL.js">
112
112
  <link rel="modulepreload" crossorigin href="./assets/arrays-DYDL-3-i.js">
113
113
  <link rel="modulepreload" crossorigin href="./assets/strings-wdPMRf6Z.js">
114
114
  <link rel="modulepreload" crossorigin href="./assets/once-DRroIaBz.js">
@@ -133,7 +133,7 @@
133
133
  <link rel="modulepreload" crossorigin href="./assets/debounce-DhnxH9Rh.js">
134
134
  <link rel="modulepreload" crossorigin href="./assets/database-zap-kIkTfzTX.js">
135
135
  <link rel="modulepreload" crossorigin href="./assets/main-B0OX4z33.js">
136
- <link rel="modulepreload" crossorigin href="./assets/cells-BP6kg4iF.js">
136
+ <link rel="modulepreload" crossorigin href="./assets/cells-BXZ3FSp9.js">
137
137
  <link rel="modulepreload" crossorigin href="./assets/ErrorBoundary-DyYDV0HI.js">
138
138
  <link rel="modulepreload" crossorigin href="./assets/kbd-CTUAEnEx.js">
139
139
  <link rel="modulepreload" crossorigin href="./assets/useInstallPackage-DUF4IRRI.js">
@@ -147,35 +147,35 @@
147
147
  <link rel="modulepreload" crossorigin href="./assets/usePress-DQ_tAz5W.js">
148
148
  <link rel="modulepreload" crossorigin href="./assets/input-C3Hrdlqq.js">
149
149
  <link rel="modulepreload" crossorigin href="./assets/ImperativeModal-B3Th7k4R.js">
150
- <link rel="modulepreload" crossorigin href="./assets/cell-link-Bkd6QnOS.js">
150
+ <link rel="modulepreload" crossorigin href="./assets/cell-link-BEtbRN19.js">
151
151
  <link rel="modulepreload" crossorigin href="./assets/multi-map-CUuNtzHt.js">
152
152
  <link rel="modulepreload" crossorigin href="./assets/alert-yTS3WpF4.js">
153
153
  <link rel="modulepreload" crossorigin href="./assets/chevron-right-CG5QYXYk.js">
154
154
  <link rel="modulepreload" crossorigin href="./assets/dropdown-menu-CR7cnzLX.js">
155
155
  <link rel="modulepreload" crossorigin href="./assets/links-D1JoyKTt.js">
156
- <link rel="modulepreload" crossorigin href="./assets/useRunCells-rD6HJnx6.js">
156
+ <link rel="modulepreload" crossorigin href="./assets/useRunCells-PwtrEJj7.js">
157
157
  <link rel="modulepreload" crossorigin href="./assets/copy-LK56fFow.js">
158
158
  <link rel="modulepreload" crossorigin href="./assets/copy-BwrPA9zQ.js">
159
159
  <link rel="modulepreload" crossorigin href="./assets/copy-icon-OjtDb4gO.js">
160
- <link rel="modulepreload" crossorigin href="./assets/RenderHTML-BvW-obUg.js">
161
- <link rel="modulepreload" crossorigin href="./assets/datasource-BjfZ5rCV.js">
162
- <link rel="modulepreload" crossorigin href="./assets/state-d_LoPKHg.js">
160
+ <link rel="modulepreload" crossorigin href="./assets/RenderHTML-CBsvWTjU.js">
161
+ <link rel="modulepreload" crossorigin href="./assets/datasource-WRQecO8l.js">
162
+ <link rel="modulepreload" crossorigin href="./assets/state-Bt-mCXoz.js">
163
163
  <link rel="modulepreload" crossorigin href="./assets/package-Tv6ztuzw.js">
164
164
  <link rel="modulepreload" crossorigin href="./assets/sparkles-lWUAsPhp.js">
165
- <link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-XPsNfvEO.js">
165
+ <link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-d8qLIXHx.js">
166
166
  <link rel="modulepreload" crossorigin href="./assets/spinner-Bhir8k53.js">
167
- <link rel="modulepreload" crossorigin href="./assets/html-to-image-EYrNFuGt.js">
168
- <link rel="modulepreload" crossorigin href="./assets/focus-D2kXD_rc.js">
167
+ <link rel="modulepreload" crossorigin href="./assets/html-to-image-Nf7ijTtQ.js">
168
+ <link rel="modulepreload" crossorigin href="./assets/focus-DDYdCBX6.js">
169
169
  <link rel="modulepreload" crossorigin href="./assets/useAsyncData-bgszE9F0.js">
170
170
  <link rel="modulepreload" crossorigin href="./assets/LazyAnyLanguageCodeMirror-BEvXb3VX.js">
171
171
  <link rel="modulepreload" crossorigin href="./assets/micromark-factory-space-BygYYKhs.js">
172
172
  <link rel="modulepreload" crossorigin href="./assets/chunk-5FQGJX7Z-D9iBG0F7.js">
173
- <link rel="modulepreload" crossorigin href="./assets/markdown-renderer-d9Lol4Mr.js">
173
+ <link rel="modulepreload" crossorigin href="./assets/markdown-renderer-C4nMzmEI.js">
174
174
  <link rel="modulepreload" crossorigin href="./assets/command-KARR7KMq.js">
175
175
  <link rel="modulepreload" crossorigin href="./assets/field-zLmMOSA4.js">
176
176
  <link rel="modulepreload" crossorigin href="./assets/popover-Bz_0Vkyf.js">
177
177
  <link rel="modulepreload" crossorigin href="./assets/errors-vr57w7Ul.js">
178
- <link rel="modulepreload" crossorigin href="./assets/download-DrC3sFE3.js">
178
+ <link rel="modulepreload" crossorigin href="./assets/download-96wP8V0p.js">
179
179
  <link rel="modulepreload" crossorigin href="./assets/table-BGPSHfig.js">
180
180
  <link rel="modulepreload" crossorigin href="./assets/useIframeCapabilities-CcI1zSdn.js">
181
181
  <link rel="modulepreload" crossorigin href="./assets/error-banner-LdWZDbqd.js">
@@ -196,40 +196,40 @@
196
196
  <link rel="modulepreload" crossorigin href="./assets/plus-BgB18UzY.js">
197
197
  <link rel="modulepreload" crossorigin href="./assets/trash-2-rVklqqFF.js">
198
198
  <link rel="modulepreload" crossorigin href="./assets/react-resizable-panels.browser.esm-CV8-hvjx.js">
199
- <link rel="modulepreload" crossorigin href="./assets/JsonOutput-DzATTBoP.js">
199
+ <link rel="modulepreload" crossorigin href="./assets/JsonOutput-H6HB87kK.js">
200
200
  <link rel="modulepreload" crossorigin href="./assets/chart-no-axes-column-nqk474t8.js">
201
201
  <link rel="modulepreload" crossorigin href="./assets/square-function-uY_yJr5g.js">
202
202
  <link rel="modulepreload" crossorigin href="./assets/spec-CPQR_o92.js">
203
203
  <link rel="modulepreload" crossorigin href="./assets/ellipsis-vertical-CkwWkOQL.js">
204
204
  <link rel="modulepreload" crossorigin href="./assets/refresh-cw-DHwG4Mac.js">
205
205
  <link rel="modulepreload" crossorigin href="./assets/tree-actions-BM_EJr3E.js">
206
- <link rel="modulepreload" crossorigin href="./assets/components-DvggA56E.js">
207
- <link rel="modulepreload" crossorigin href="./assets/column-preview-9_vf_vQt.js">
206
+ <link rel="modulepreload" crossorigin href="./assets/components-CpiEO4PD.js">
207
+ <link rel="modulepreload" crossorigin href="./assets/column-preview-DDR-SBCD.js">
208
208
  <link rel="modulepreload" crossorigin href="./assets/icons-Ol38nIbL.js">
209
- <link rel="modulepreload" crossorigin href="./assets/radio-group-cvI-M2aa.js">
210
- <link rel="modulepreload" crossorigin href="./assets/floating-outline-CDjyi71K.js">
209
+ <link rel="modulepreload" crossorigin href="./assets/radio-group-BgIE4hAy.js">
210
+ <link rel="modulepreload" crossorigin href="./assets/floating-outline-Cvm2qN8K.js">
211
211
  <link rel="modulepreload" crossorigin href="./assets/objectWithoutPropertiesLoose-DfWeGRFv.js">
212
212
  <link rel="modulepreload" crossorigin href="./assets/esm-CqWdmSnV.js">
213
- <link rel="modulepreload" crossorigin href="./assets/readonly-python-code-BmZ0KZKm.js">
213
+ <link rel="modulepreload" crossorigin href="./assets/readonly-python-code-Cmj3Lveo.js">
214
214
  <link rel="modulepreload" crossorigin href="./assets/file-headphone-BrQspHac.js">
215
215
  <link rel="modulepreload" crossorigin href="./assets/file-BrdxGLRX.js">
216
216
  <link rel="modulepreload" crossorigin href="./assets/image-DQHXdEQn.js">
217
- <link rel="modulepreload" crossorigin href="./assets/file-icons-pUZ3DUy5.js">
218
- <link rel="modulepreload" crossorigin href="./assets/switch-YcZLs0mS.js">
217
+ <link rel="modulepreload" crossorigin href="./assets/file-icons-CVihyOvq.js">
218
+ <link rel="modulepreload" crossorigin href="./assets/switch-Z7gfNPvp.js">
219
219
  <link rel="modulepreload" crossorigin href="./assets/events-CPoJAfgx.js">
220
- <link rel="modulepreload" crossorigin href="./assets/globals-Dd7eo1nD.js">
220
+ <link rel="modulepreload" crossorigin href="./assets/globals-CYbHM4QS.js">
221
221
  <link rel="modulepreload" crossorigin href="./assets/share-DMwTZOTH.js">
222
222
  <link rel="modulepreload" crossorigin href="./assets/blob-Bgnx1kuY.js">
223
223
  <link rel="modulepreload" crossorigin href="./assets/memoize-Tp7rARFe.js">
224
224
  <link rel="modulepreload" crossorigin href="./assets/get-C-qh_et5.js">
225
225
  <link rel="modulepreload" crossorigin href="./assets/_baseSet-CxV9N1bc.js">
226
- <link rel="modulepreload" crossorigin href="./assets/state-G8Xj60GT.js">
226
+ <link rel="modulepreload" crossorigin href="./assets/state-DUrsoPal.js">
227
227
  <link rel="modulepreload" crossorigin href="./assets/label-xHqFtfdz.js">
228
- <link rel="modulepreload" crossorigin href="./assets/textarea-DV6LIVoz.js">
228
+ <link rel="modulepreload" crossorigin href="./assets/textarea-CMdRAIqd.js">
229
229
  <link rel="modulepreload" crossorigin href="./assets/refresh-ccw-C-n2VFP5.js">
230
- <link rel="modulepreload" crossorigin href="./assets/form-COQrUWdR.js">
231
- <link rel="modulepreload" crossorigin href="./assets/renderShortcut-CzUfgpdJ.js">
232
- <link rel="modulepreload" crossorigin href="./assets/useBoolean-oqDq5zze.js">
230
+ <link rel="modulepreload" crossorigin href="./assets/form-DLzm3BCq.js">
231
+ <link rel="modulepreload" crossorigin href="./assets/renderShortcut-CUc_HCHW.js">
232
+ <link rel="modulepreload" crossorigin href="./assets/useBoolean-Begogf9u.js">
233
233
  <link rel="modulepreload" crossorigin href="./assets/useDeepCompareMemoize-zUHU--0D.js">
234
234
  <link rel="modulepreload" crossorigin href="./assets/types-W8WWuumF.js">
235
235
  <link rel="modulepreload" crossorigin href="./assets/semaphore-FlZezxaf.js">
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@marimo-team/frontend",
3
- "version": "0.23.9-dev34",
3
+ "version": "0.23.9-dev36",
4
4
  "main": "dist/main.js",
5
5
  "types": "dist/index.d.ts",
6
6
  "type": "module",
@@ -1849,6 +1849,38 @@ export type AiSettingsSubTab =
1849
1849
  | "ai-models"
1850
1850
  | "mcp";
1851
1851
 
1852
+ const AiEnabledConfig: React.FC<AiConfigProps> = ({ form, config }) => {
1853
+ return (
1854
+ <SettingGroup>
1855
+ <FormField
1856
+ control={form.control}
1857
+ name="ai.enabled"
1858
+ render={({ field }) => (
1859
+ <div className="flex flex-col gap-y-1">
1860
+ <FormItem className={formItemClasses}>
1861
+ <FormLabel className="font-normal">Enable AI features</FormLabel>
1862
+ <FormControl>
1863
+ <Checkbox
1864
+ data-testid="ai-enabled-checkbox"
1865
+ checked={field.value !== false}
1866
+ onCheckedChange={(checked) =>
1867
+ field.onChange(checked === true)
1868
+ }
1869
+ />
1870
+ </FormControl>
1871
+ <IsOverridden userConfig={config} name="ai.enabled" />
1872
+ </FormItem>
1873
+ <FormDescription>
1874
+ When disabled, AI actions and panels are hidden from the marimo
1875
+ UI.
1876
+ </FormDescription>
1877
+ </div>
1878
+ )}
1879
+ />
1880
+ </SettingGroup>
1881
+ );
1882
+ };
1883
+
1852
1884
  export const AiConfig: React.FC<AiConfigProps> = ({
1853
1885
  form,
1854
1886
  config,
@@ -1857,18 +1889,30 @@ export const AiConfig: React.FC<AiConfigProps> = ({
1857
1889
  // MCP is not supported in WASM
1858
1890
  const wasm = isWasm();
1859
1891
  const [activeTab, setActiveTab] = useAtom(aiSettingsSubTabAtom);
1892
+ const aiEnabled = useWatch({
1893
+ control: form.control,
1894
+ name: "ai.enabled",
1895
+ });
1896
+ const activeVisibleTab =
1897
+ aiEnabled === false && activeTab !== "ai-features"
1898
+ ? "ai-features"
1899
+ : activeTab;
1860
1900
 
1861
1901
  return (
1862
1902
  <Tabs
1863
- value={activeTab}
1903
+ value={activeVisibleTab}
1864
1904
  onValueChange={(value) => setActiveTab(value as AiSettingsSubTab)}
1865
1905
  className="flex-1"
1866
1906
  >
1867
1907
  <TabsList className="mb-2">
1868
1908
  <TabsTrigger value="ai-features">AI Features</TabsTrigger>
1869
- <TabsTrigger value="ai-providers">AI Providers</TabsTrigger>
1870
- <TabsTrigger value="ai-models">AI Models</TabsTrigger>
1871
- {!wasm && <TabsTrigger value="mcp">MCP</TabsTrigger>}
1909
+ {aiEnabled !== false && (
1910
+ <>
1911
+ <TabsTrigger value="ai-providers">AI Providers</TabsTrigger>
1912
+ <TabsTrigger value="ai-models">AI Models</TabsTrigger>
1913
+ {!wasm && <TabsTrigger value="mcp">MCP</TabsTrigger>}
1914
+ </>
1915
+ )}
1872
1916
  </TabsList>
1873
1917
 
1874
1918
  <TabsContent value="ai-features">
@@ -1877,18 +1921,33 @@ export const AiConfig: React.FC<AiConfigProps> = ({
1877
1921
  config={config}
1878
1922
  onSubmit={onSubmit}
1879
1923
  />
1880
- <AiAssistConfig form={form} config={config} onSubmit={onSubmit} />
1881
- </TabsContent>
1882
- <TabsContent value="ai-providers">
1883
- <AiProvidersConfig form={form} config={config} onSubmit={onSubmit} />
1884
- </TabsContent>
1885
- <TabsContent value="ai-models">
1886
- <AiModelDisplayConfig form={form} config={config} onSubmit={onSubmit} />
1924
+ <AiEnabledConfig form={form} config={config} onSubmit={onSubmit} />
1925
+ {aiEnabled !== false && (
1926
+ <AiAssistConfig form={form} config={config} onSubmit={onSubmit} />
1927
+ )}
1887
1928
  </TabsContent>
1888
- {!wasm && (
1889
- <TabsContent value="mcp">
1890
- <MCPConfig form={form} onSubmit={onSubmit} />
1891
- </TabsContent>
1929
+ {aiEnabled !== false && (
1930
+ <>
1931
+ <TabsContent value="ai-providers">
1932
+ <AiProvidersConfig
1933
+ form={form}
1934
+ config={config}
1935
+ onSubmit={onSubmit}
1936
+ />
1937
+ </TabsContent>
1938
+ <TabsContent value="ai-models">
1939
+ <AiModelDisplayConfig
1940
+ form={form}
1941
+ config={config}
1942
+ onSubmit={onSubmit}
1943
+ />
1944
+ </TabsContent>
1945
+ {!wasm && (
1946
+ <TabsContent value="mcp">
1947
+ <MCPConfig form={form} onSubmit={onSubmit} />
1948
+ </TabsContent>
1949
+ )}
1950
+ </>
1892
1951
  )}
1893
1952
  </Tabs>
1894
1953
  );
@@ -51,7 +51,7 @@ import {
51
51
  FRONTEND_TOOL_REGISTRY,
52
52
  } from "@/core/ai/tools/registry";
53
53
  import { useCellActions } from "@/core/cells/cells";
54
- import { aiAtom, aiEnabledAtom } from "@/core/config/config";
54
+ import { aiAtom, aiModelConfiguredAtom } from "@/core/config/config";
55
55
  import { DEFAULT_AI_MODEL } from "@/core/config/config-schema";
56
56
  import { useRequestClient } from "@/core/network/requests";
57
57
  import { useRuntimeManager } from "@/core/runtime/config";
@@ -442,7 +442,7 @@ const PairWithAgentCallout: React.FC<{
442
442
  };
443
443
 
444
444
  const ChatPanel = () => {
445
- const aiConfigured = useAtomValue(aiEnabledAtom);
445
+ const aiConfigured = useAtomValue(aiModelConfiguredAtom);
446
446
  const { handleClick } = useOpenSettingsToTab();
447
447
 
448
448
  if (!aiConfigured) {
@@ -1,6 +1,11 @@
1
1
  /* Copyright 2026 Marimo. All rights reserved. */
2
2
 
3
- import type { Column, SortingState } from "@tanstack/react-table";
3
+ import type {
4
+ Column,
5
+ SortDirection,
6
+ SortingState,
7
+ Table,
8
+ } from "@tanstack/react-table";
4
9
  import { fireEvent, render, screen } from "@testing-library/react";
5
10
  import { describe, expect, it, vi } from "vitest";
6
11
  import {
@@ -8,7 +13,23 @@ import {
8
13
  DropdownMenuContent,
9
14
  DropdownMenuTrigger,
10
15
  } from "@/components/ui/dropdown-menu";
11
- import { HideColumn } from "../header-items";
16
+ import {
17
+ ColumnPinning,
18
+ ColumnWrapping,
19
+ CopyColumn,
20
+ DataType,
21
+ FormatOptions,
22
+ HideColumn,
23
+ Sorts,
24
+ } from "../header-items";
25
+
26
+ const renderInMenu = (node: React.ReactNode) =>
27
+ render(
28
+ <DropdownMenu open={true}>
29
+ <DropdownMenuTrigger />
30
+ <DropdownMenuContent>{node}</DropdownMenuContent>
31
+ </DropdownMenu>,
32
+ );
12
33
 
13
34
  describe("multi-column sorting logic", () => {
14
35
  // Extract the core sorting logic to test in isolation
@@ -167,14 +188,6 @@ describe("HideColumn", () => {
167
188
  toggleVisibility,
168
189
  }) as unknown as Column<unknown, unknown>;
169
190
 
170
- const renderInMenu = (node: React.ReactNode) =>
171
- render(
172
- <DropdownMenu open={true}>
173
- <DropdownMenuTrigger />
174
- <DropdownMenuContent>{node}</DropdownMenuContent>
175
- </DropdownMenu>,
176
- );
177
-
178
191
  it("renders 'Hide column' when canHide is true", () => {
179
192
  renderInMenu(<HideColumn column={makeColumn()} />);
180
193
  expect(screen.getByText("Hide column")).toBeInTheDocument();
@@ -192,3 +205,200 @@ describe("HideColumn", () => {
192
205
  expect(toggleVisibility).toHaveBeenCalledWith(false);
193
206
  });
194
207
  });
208
+
209
+ describe("DataType", () => {
210
+ const makeColumn = (dtype?: string) =>
211
+ ({
212
+ columnDef: { meta: dtype === undefined ? {} : { dtype } },
213
+ }) as unknown as Column<unknown, unknown>;
214
+
215
+ it("renders the dtype label when present", () => {
216
+ renderInMenu(<DataType column={makeColumn("int64")} />);
217
+ expect(screen.getByText("int64")).toBeInTheDocument();
218
+ });
219
+
220
+ it("returns null when dtype is absent", () => {
221
+ renderInMenu(<DataType column={makeColumn()} />);
222
+ expect(screen.queryByText("int64")).toBeNull();
223
+ });
224
+ });
225
+
226
+ describe("Sorts", () => {
227
+ const makeColumn = ({
228
+ canSort = true,
229
+ sorted = false,
230
+ sortIndex = 0,
231
+ }: {
232
+ canSort?: boolean;
233
+ sorted?: false | SortDirection;
234
+ sortIndex?: number;
235
+ } = {}) =>
236
+ ({
237
+ getCanSort: () => canSort,
238
+ getIsSorted: () => sorted,
239
+ getSortIndex: () => sortIndex,
240
+ clearSorting: vi.fn(),
241
+ toggleSorting: vi.fn(),
242
+ }) as unknown as Column<unknown, unknown>;
243
+
244
+ const makeTable = (sorting: SortingState) =>
245
+ ({
246
+ getState: () => ({ sorting }),
247
+ resetSorting: vi.fn(),
248
+ }) as unknown as Table<unknown>;
249
+
250
+ it("returns null when the column cannot sort", () => {
251
+ renderInMenu(<Sorts column={makeColumn({ canSort: false })} />);
252
+ expect(screen.queryByText("Asc")).toBeNull();
253
+ });
254
+
255
+ it("renders Asc and Desc items", () => {
256
+ renderInMenu(<Sorts column={makeColumn()} />);
257
+ expect(screen.getByText("Asc")).toBeInTheDocument();
258
+ expect(screen.getByText("Desc")).toBeInTheDocument();
259
+ });
260
+
261
+ it("offers single-column 'Clear sort' when sorted without multi-sort", () => {
262
+ renderInMenu(<Sorts column={makeColumn({ sorted: "asc" })} />);
263
+ expect(screen.getByText("Clear sort")).toBeInTheDocument();
264
+ });
265
+
266
+ it("offers 'Clear all sorts' when the table has multiple sorts", () => {
267
+ renderInMenu(
268
+ <Sorts
269
+ column={makeColumn({ sorted: "asc" })}
270
+ table={makeTable([
271
+ { id: "a", desc: false },
272
+ { id: "b", desc: true },
273
+ ])}
274
+ />,
275
+ );
276
+ expect(screen.getByText("Clear all sorts")).toBeInTheDocument();
277
+ });
278
+ });
279
+
280
+ describe("CopyColumn", () => {
281
+ const makeColumn = ({
282
+ canCopy = true,
283
+ id = "name",
284
+ }: {
285
+ canCopy?: boolean;
286
+ id?: string;
287
+ } = {}) =>
288
+ ({
289
+ id,
290
+ getCanCopy: () => canCopy,
291
+ }) as unknown as Column<unknown, unknown>;
292
+
293
+ it("renders 'Copy column name' when copyable", () => {
294
+ renderInMenu(<CopyColumn column={makeColumn()} />);
295
+ expect(screen.getByText("Copy column name")).toBeInTheDocument();
296
+ });
297
+
298
+ it("returns null when the column cannot be copied", () => {
299
+ renderInMenu(<CopyColumn column={makeColumn({ canCopy: false })} />);
300
+ expect(screen.queryByText("Copy column name")).toBeNull();
301
+ });
302
+ });
303
+
304
+ describe("ColumnPinning", () => {
305
+ const makeColumn = ({
306
+ canPin = true,
307
+ pinned = false,
308
+ }: {
309
+ canPin?: boolean;
310
+ pinned?: false | "left" | "right";
311
+ } = {}) =>
312
+ ({
313
+ getCanPin: () => canPin,
314
+ getIsPinned: () => pinned,
315
+ pin: vi.fn(),
316
+ }) as unknown as Column<unknown, unknown>;
317
+
318
+ it("returns null when the column cannot be pinned", () => {
319
+ renderInMenu(<ColumnPinning column={makeColumn({ canPin: false })} />);
320
+ expect(screen.queryByText("Freeze left")).toBeNull();
321
+ });
322
+
323
+ it("offers freeze options when unpinned", () => {
324
+ renderInMenu(<ColumnPinning column={makeColumn()} />);
325
+ expect(screen.getByText("Freeze left")).toBeInTheDocument();
326
+ expect(screen.getByText("Freeze right")).toBeInTheDocument();
327
+ });
328
+
329
+ it("offers 'Unfreeze' when pinned", () => {
330
+ renderInMenu(<ColumnPinning column={makeColumn({ pinned: "left" })} />);
331
+ expect(screen.getByText("Unfreeze")).toBeInTheDocument();
332
+ });
333
+ });
334
+
335
+ describe("ColumnWrapping", () => {
336
+ const makeColumn = ({
337
+ canWrap = true,
338
+ wrapping = "nowrap",
339
+ }: {
340
+ canWrap?: boolean;
341
+ wrapping?: "wrap" | "nowrap";
342
+ } = {}) =>
343
+ ({
344
+ getCanWrap: () => canWrap,
345
+ getColumnWrapping: () => wrapping,
346
+ toggleColumnWrapping: vi.fn(),
347
+ }) as unknown as Column<unknown, unknown>;
348
+
349
+ it("returns null when the column cannot wrap", () => {
350
+ renderInMenu(<ColumnWrapping column={makeColumn({ canWrap: false })} />);
351
+ expect(screen.queryByText("Wrap text")).toBeNull();
352
+ });
353
+
354
+ it("offers 'Wrap text' when not wrapping", () => {
355
+ renderInMenu(<ColumnWrapping column={makeColumn()} />);
356
+ expect(screen.getByText("Wrap text")).toBeInTheDocument();
357
+ });
358
+
359
+ it("offers 'No wrap text' when wrapping", () => {
360
+ renderInMenu(<ColumnWrapping column={makeColumn({ wrapping: "wrap" })} />);
361
+ expect(screen.getByText("No wrap text")).toBeInTheDocument();
362
+ });
363
+ });
364
+
365
+ describe("FormatOptions", () => {
366
+ const makeColumn = ({
367
+ dataType = "number",
368
+ canFormat = true,
369
+ }: {
370
+ dataType?: string;
371
+ canFormat?: boolean;
372
+ } = {}) =>
373
+ ({
374
+ columnDef: { meta: { dataType } },
375
+ getCanFormat: () => canFormat,
376
+ getColumnFormatting: () => undefined,
377
+ setColumnFormatting: vi.fn(),
378
+ }) as unknown as Column<unknown, unknown>;
379
+
380
+ it("renders the 'Format' submenu trigger for formattable columns", () => {
381
+ renderInMenu(<FormatOptions column={makeColumn()} locale="en-US" />);
382
+ expect(screen.getByText("Format")).toBeInTheDocument();
383
+ });
384
+
385
+ it("returns null when the column cannot be formatted", () => {
386
+ renderInMenu(
387
+ <FormatOptions
388
+ column={makeColumn({ canFormat: false })}
389
+ locale="en-US"
390
+ />,
391
+ );
392
+ expect(screen.queryByText("Format")).toBeNull();
393
+ });
394
+
395
+ it("returns null when the data type has no format options", () => {
396
+ renderInMenu(
397
+ <FormatOptions
398
+ column={makeColumn({ dataType: "unknown" })}
399
+ locale="en-US"
400
+ />,
401
+ );
402
+ expect(screen.queryByText("Format")).toBeNull();
403
+ });
404
+ });