@marimo-team/frontend 0.23.10-dev1 → 0.23.10-dev13

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 (97) hide show
  1. package/dist/assets/{CellStatus-BLWR83i_.js → CellStatus-BNpTS2Th.js} +1 -1
  2. package/dist/assets/{JsonOutput-BJaeXQim.js → JsonOutput-CPYntVn7.js} +12 -12
  3. package/dist/assets/{LazyAnyLanguageCodeMirror-BEvXb3VX.js → LazyAnyLanguageCodeMirror-DXGWNDPu.js} +2 -2
  4. package/dist/assets/{MarimoErrorOutput-B0h5OVW9.js → MarimoErrorOutput-D8_FYh4M.js} +1 -1
  5. package/dist/assets/{RenderHTML-ChwPJgPd.js → RenderHTML-BlI-DkEC.js} +1 -1
  6. package/dist/assets/{add-cell-with-ai-FxJitVGs.js → add-cell-with-ai-BLPYuDtc.js} +1 -1
  7. package/dist/assets/{add-connection-dialog-EgkKupsZ.js → add-connection-dialog-BUOUhMLU.js} +1 -1
  8. package/dist/assets/{agent-panel-D5uoss1C.js → agent-panel-BWh5ASpI.js} +1 -1
  9. package/dist/assets/{ai-model-dropdown-DxAk02Lc.js → ai-model-dropdown-WfztO91E.js} +1 -1
  10. package/dist/assets/{any-language-editor-CXeLf_N8.js → any-language-editor-DOLtgCLo.js} +1 -1
  11. package/dist/assets/{app-config-button-BcOcIm9W.js → app-config-button-BhsBqwXH.js} +1 -1
  12. package/dist/assets/{cache-panel-DYiVMv5x.js → cache-panel-lRBWNjiY.js} +1 -1
  13. package/dist/assets/{cell-editor-C5CDtKML.js → cell-editor-BSGwHnNC.js} +1 -1
  14. package/dist/assets/{cell-link-B1pHtGWp.js → cell-link-CB8WeUi5.js} +1 -1
  15. package/dist/assets/{cells-BsaJrRWi.js → cells-DrYr16TO.js} +63 -63
  16. package/dist/assets/{chat-display-ccjE30z9.js → chat-display-DKS2riey.js} +1 -1
  17. package/dist/assets/{chat-panel-CtvtP3kb.js → chat-panel-B2g8DKMb.js} +1 -1
  18. package/dist/assets/{chat-ui-DV8CD_s1.js → chat-ui-CoZhAigw.js} +1 -1
  19. package/dist/assets/{column-preview-BPismotx.js → column-preview-B16EcXyK.js} +1 -1
  20. package/dist/assets/{command-palette-Cdy8eCpi.js → command-palette-BDdSMeT0.js} +1 -1
  21. package/dist/assets/{common-6n8M1qiK.js → common-zEvAGAwj.js} +1 -1
  22. package/dist/assets/{components-BzBUxUz0.js → components-CB5nynlN.js} +1 -1
  23. package/dist/assets/{components-CK-WtvAJ.js → components-LGou1UYG.js} +1 -1
  24. package/dist/assets/{copy-icon-OjtDb4gO.js → copy-icon-D_qz1txR.js} +1 -1
  25. package/dist/assets/{datasource-DjDpJnbt.js → datasource-BKe0SLTo.js} +1 -1
  26. package/dist/assets/{dependency-graph-panel-BNaXAcZr.js → dependency-graph-panel-C2rUhrcE.js} +1 -1
  27. package/dist/assets/{documentation-panel-5jUKZR-l.js → documentation-panel-cQaGwB7v.js} +1 -1
  28. package/dist/assets/{download-DT8gf0PL.js → download-DZUw3TyK.js} +1 -1
  29. package/dist/assets/{edit-page-C-qF9lwB.js → edit-page-xztDGvY6.js} +6 -6
  30. package/dist/assets/{error-panel-DHcg_ER5.js → error-panel-D3mqrmOy.js} +1 -1
  31. package/dist/assets/{file-explorer-panel-CbOdvGyZ.js → file-explorer-panel-DeEwvxtD.js} +1 -1
  32. package/dist/assets/{file-icons-BiMOJy0r.js → file-icons-C8WXgGhY.js} +1 -1
  33. package/dist/assets/{file-name-input-BNHbCr1E.js → file-name-input-BQHsY06u.js} +1 -1
  34. package/dist/assets/{floating-outline-BN9sAcQ0.js → floating-outline-3J8SCP2i.js} +1 -1
  35. package/dist/assets/{focus-q38vUCl-.js → focus-1NHETejS.js} +1 -1
  36. package/dist/assets/{form-BtBTvVVd.js → form-66bEiwKr.js} +1 -1
  37. package/dist/assets/{glide-data-editor-DqKNgyfg.js → glide-data-editor-k70Tmujb.js} +1 -1
  38. package/dist/assets/{home-page-lXjPDNM0.js → home-page-C5b2fGpl.js} +1 -1
  39. package/dist/assets/{hooks-DP41kAWj.js → hooks-C14Ebau9.js} +1 -1
  40. package/dist/assets/{html-to-image-BWL3Ct4n.js → html-to-image-C9fYqzra.js} +1 -1
  41. package/dist/assets/{index-CE1XkZU_.js → index-BQp8477C.js} +16 -16
  42. package/dist/assets/{kiosk-mode-BqvqWsFx.js → kiosk-mode-e_ceeRFY.js} +1 -1
  43. package/dist/assets/{layout-DXDnDtNw.js → layout-DaBkZvqU.js} +3 -3
  44. package/dist/assets/{logs-panel-BNoBKY1q.js → logs-panel-C1ViUa_T.js} +1 -1
  45. package/dist/assets/{markdown-renderer-CU5Lbyax.js → markdown-renderer-DY7HNEoY.js} +1 -1
  46. package/dist/assets/{name-cell-input-BNc3X9qF.js → name-cell-input-DMOCXKaK.js} +1 -1
  47. package/dist/assets/{outline-panel-De2gw3Ro.js → outline-panel-Ck8BQkF3.js} +1 -1
  48. package/dist/assets/{packages-panel-C8maQt0Q.js → packages-panel-a9lhnFOg.js} +1 -1
  49. package/dist/assets/{panels-BKv_j2lC.js → panels-DRxv7U4C.js} +1 -1
  50. package/dist/assets/{process-output-B2oklQi4.js → process-output-BswD2k4M.js} +1 -1
  51. package/dist/assets/{radio-group-DlVSuZ_1.js → radio-group-2AaAWNrb.js} +1 -1
  52. package/dist/assets/{readonly-python-code-D3_kHsDx.js → readonly-python-code-8smCAF83.js} +1 -1
  53. package/dist/assets/{reveal-component-C50CTy8h.js → reveal-component-D-_WBPSy.js} +1 -1
  54. package/dist/assets/{run-page-DeOIw0On.js → run-page-Dbr9JuQ-.js} +1 -1
  55. package/dist/assets/{scratchpad-panel-DWbD_QB3.js → scratchpad-panel-D_ImFN9n.js} +1 -1
  56. package/dist/assets/{secrets-panel-BBvdIhtU.js → secrets-panel-Yt3orrjq.js} +1 -1
  57. package/dist/assets/{session-panel-BHgL-7KC.js → session-panel-BpIqcHC5.js} +1 -1
  58. package/dist/assets/{share-DMwTZOTH.js → share-mSf5FZIP.js} +1 -1
  59. package/dist/assets/{snippets-panel-FY_Ezibl.js → snippets-panel-Dew8dvfp.js} +1 -1
  60. package/dist/assets/{state-D-2KaAHK.js → state-DttgoaV8.js} +1 -1
  61. package/dist/assets/{state-BTTshiKq.js → state-ZFlMDeHg.js} +1 -1
  62. package/dist/assets/{textarea-DwltG8V6.js → textarea-Cxjg9eKY.js} +1 -1
  63. package/dist/assets/{tracing-zLH-qWb2.js → tracing-Dt4NocdT.js} +1 -1
  64. package/dist/assets/{tracing-panel-bN0O3PuU.js → tracing-panel-79O4yC1b.js} +2 -2
  65. package/dist/assets/use-toast-CJ1Tj2eC.js +1 -0
  66. package/dist/assets/{useCellActionButton-h5LjhtEx.js → useCellActionButton-eohdjp8c.js} +1 -1
  67. package/dist/assets/{useDeleteCell-BBPIhnQj.js → useDeleteCell-C43PMWBl.js} +1 -1
  68. package/dist/assets/{useDependencyPanelTab-BSkmlMqr.js → useDependencyPanelTab-BfzEwyF5.js} +1 -1
  69. package/dist/assets/{useInstallPackage-DUF4IRRI.js → useInstallPackage-GI6gu7_S.js} +1 -1
  70. package/dist/assets/{useNotebookActions-B32eCmtG.js → useNotebookActions-BKdPPMYL.js} +1 -1
  71. package/dist/assets/{useRunCells-BiCBGyeW.js → useRunCells-B7Ie4Pwb.js} +1 -1
  72. package/dist/assets/{useSplitCell-uogO-7ac.js → useSplitCell-BJAP3FSl.js} +1 -1
  73. package/dist/assets/{write-secret-modal-BFLlHwgU.js → write-secret-modal-CLc8TNUL.js} +1 -1
  74. package/dist/index.html +27 -27
  75. package/package.json +2 -2
  76. package/src/components/data-table/__tests__/data-table.test.tsx +154 -12
  77. package/src/components/data-table/hover-tooltip/__tests__/content.test.ts +60 -0
  78. package/src/components/data-table/hover-tooltip/content.ts +44 -0
  79. package/src/components/data-table/hover-tooltip/hover-tooltip.tsx +55 -0
  80. package/src/components/data-table/hover-tooltip/use-table-hover-tooltip.ts +159 -0
  81. package/src/components/data-table/renderers.tsx +27 -43
  82. package/src/components/editor/cell/cell-context-menu.tsx +15 -2
  83. package/src/components/ui/__tests__/use-toast.test.ts +75 -0
  84. package/src/components/ui/use-toast.ts +33 -13
  85. package/src/core/cells/__tests__/__snapshots__/cells.test.ts.snap +0 -28
  86. package/src/core/cells/__tests__/cell.test.ts +29 -2
  87. package/src/core/cells/__tests__/document-changes.test.ts +201 -2
  88. package/src/core/cells/cell.ts +5 -1
  89. package/src/core/cells/document-changes.ts +61 -3
  90. package/src/core/codemirror/go-to-definition/__tests__/utils.test.ts +37 -0
  91. package/src/core/codemirror/go-to-definition/commands.ts +17 -9
  92. package/src/core/codemirror/go-to-definition/utils.ts +1 -0
  93. package/src/core/codemirror/language/languages/sql/utils.ts +3 -1
  94. package/src/core/network/__tests__/requests-static.test.ts +30 -0
  95. package/src/core/network/requests-static.ts +14 -10
  96. package/src/plugins/layout/DownloadPlugin.tsx +1 -1
  97. package/dist/assets/use-toast-IPAtJGea.js +0 -1
@@ -1 +1 @@
1
- import{s as V}from"./chunk-LvLJmgfZ.js";import{t as B}from"./react-Bj1aDYRI.js";import{t as G}from"./compiler-runtime-B3qBwwSJ.js";import{t as H}from"./jsx-runtime-BqBOg78p.js";import{c as I,i as J,n as O,s as Q,t as U}from"./select-DZcFyKFQ.js";import{t as P}from"./label-xHqFtfdz.js";import{t as W}from"./button-BbCh-29a.js";import{r as X}from"./requests-DIwGYs0l.js";import{r as D}from"./input-C3Hrdlqq.js";import{t as E}from"./use-toast-IPAtJGea.js";import{a as Z,c as $,i as ee,n as te,r as re}from"./dialog-DzC_QCtT.js";import{t as ae}from"./links-D1JoyKTt.js";import{r as L}from"./field-zLmMOSA4.js";var oe=G(),T=V(B(),1),t=V(H(),1);function ne(a){return a.toSorted((e,r)=>e.provider==="env"?1:r.provider==="env"?-1:0)}const le=a=>{let e=(0,oe.c)(43),{providerNames:r,onClose:A,onSuccess:F}=a,{writeSecret:Y}=X(),[o,M]=T.useState(""),[l,R]=T.useState(""),[n,z]=T.useState(r[0]),g;e[0]!==o||e[1]!==n||e[2]!==F||e[3]!==l||e[4]!==Y?(g=async i=>{if(i.preventDefault(),!n){E({title:"Error",description:"No location selected for the secret.",variant:"danger"});return}if(!o||!l||!n){E({title:"Error",description:"Please fill in all fields.",variant:"danger"});return}try{await Y({key:o,value:l,provider:"dotenv",name:n}),E({title:"Secret created",description:"The secret has been created successfully."}),F(o)}catch{E({title:"Error",description:"Failed to create secret. Please try again.",variant:"danger"})}},e[0]=o,e[1]=n,e[2]=F,e[3]=l,e[4]=Y,e[5]=g):g=e[5];let q=g,j;e[6]===Symbol.for("react.memo_cache_sentinel")?(j=(0,t.jsxs)(Z,{children:[(0,t.jsx)($,{children:"Add Secret"}),(0,t.jsx)(re,{children:"Add a new secret to your environment variables."})]}),e[6]=j):j=e[6];let y;e[7]===Symbol.for("react.memo_cache_sentinel")?(y=(0,t.jsx)(P,{htmlFor:"key",children:"Key"}),e[7]=y):y=e[7];let S;e[8]===Symbol.for("react.memo_cache_sentinel")?(S=i=>{M(ie(i.target.value))},e[8]=S):S=e[8];let s;e[9]===o?s=e[10]:(s=(0,t.jsxs)("div",{className:"grid gap-2",children:[y,(0,t.jsx)(D,{id:"key",value:o,onChange:S,placeholder:"MY_SECRET_KEY",required:!0})]}),e[9]=o,e[10]=s);let _;e[11]===Symbol.for("react.memo_cache_sentinel")?(_=(0,t.jsx)(P,{htmlFor:"value",children:"Value"}),e[11]=_):_=e[11];let b;e[12]===Symbol.for("react.memo_cache_sentinel")?(b=i=>R(i.target.value),e[12]=b):b=e[12];let c;e[13]===l?c=e[14]:(c=(0,t.jsx)(D,{id:"value",type:"password",value:l,onChange:b,required:!0,autoComplete:"off"}),e[13]=l,e[14]=c);let C;e[15]===Symbol.for("react.memo_cache_sentinel")?(C=se()&&(0,t.jsx)(L,{children:"Note: You are sending this key over http."}),e[15]=C):C=e[15];let d;e[16]===c?d=e[17]:(d=(0,t.jsxs)("div",{className:"grid gap-2",children:[_,c,C]}),e[16]=c,e[17]=d);let N;e[18]===Symbol.for("react.memo_cache_sentinel")?(N=(0,t.jsx)(P,{htmlFor:"location",children:"Location"}),e[18]=N):N=e[18];let m;e[19]===r.length?m=e[20]:(m=r.length===0&&(0,t.jsx)("p",{className:"text-sm text-muted-foreground",children:"No dotenv locations configured."}),e[19]=r.length,e[20]=m);let h;e[21]!==n||e[22]!==r?(h=r.length>0&&(0,t.jsxs)(U,{value:n,onValueChange:i=>z(i),children:[(0,t.jsx)(Q,{children:(0,t.jsx)(I,{placeholder:"Select a provider"})}),(0,t.jsx)(O,{children:r.map(ce)})]}),e[21]=n,e[22]=r,e[23]=h):h=e[23];let k;e[24]===Symbol.for("react.memo_cache_sentinel")?(k=(0,t.jsxs)(L,{children:["You can configure the location by setting the"," ",(0,t.jsx)(ae,{href:"https://links.marimo.app/dotenv",children:"dotenv configuration"}),"."]}),e[24]=k):k=e[24];let p;e[25]!==m||e[26]!==h?(p=(0,t.jsxs)("div",{className:"grid gap-2",children:[N,m,h,k]}),e[25]=m,e[26]=h,e[27]=p):p=e[27];let u;e[28]!==d||e[29]!==p||e[30]!==s?(u=(0,t.jsxs)("div",{className:"grid gap-4 py-4",children:[s,d,p]}),e[28]=d,e[29]=p,e[30]=s,e[31]=u):u=e[31];let f;e[32]===A?f=e[33]:(f=(0,t.jsx)(W,{type:"button",variant:"outline",onClick:A,children:"Cancel"}),e[32]=A,e[33]=f);let K=!o||!l||!n,v;e[34]===K?v=e[35]:(v=(0,t.jsx)(W,{type:"submit",disabled:K,children:"Add Secret"}),e[34]=K,e[35]=v);let x;e[36]!==f||e[37]!==v?(x=(0,t.jsxs)(ee,{children:[f,v]}),e[36]=f,e[37]=v,e[38]=x):x=e[38];let w;return e[39]!==q||e[40]!==u||e[41]!==x?(w=(0,t.jsx)(te,{children:(0,t.jsxs)("form",{onSubmit:q,children:[j,u,x]})}),e[39]=q,e[40]=u,e[41]=x,e[42]=w):w=e[42],w};function ie(a){return a.replaceAll(/\W/g,"_")}function se(){return window.location.href.startsWith("http://")}function ce(a){return(0,t.jsx)(J,{value:a,children:a},a)}export{ne as n,le as t};
1
+ import{s as V}from"./chunk-LvLJmgfZ.js";import{t as B}from"./react-Bj1aDYRI.js";import{t as G}from"./compiler-runtime-B3qBwwSJ.js";import{t as H}from"./jsx-runtime-BqBOg78p.js";import{c as I,i as J,n as O,s as Q,t as U}from"./select-DZcFyKFQ.js";import{t as P}from"./label-xHqFtfdz.js";import{t as W}from"./button-BbCh-29a.js";import{r as X}from"./requests-DIwGYs0l.js";import{r as D}from"./input-C3Hrdlqq.js";import{t as E}from"./use-toast-CJ1Tj2eC.js";import{a as Z,c as $,i as ee,n as te,r as re}from"./dialog-DzC_QCtT.js";import{t as ae}from"./links-D1JoyKTt.js";import{r as L}from"./field-zLmMOSA4.js";var oe=G(),T=V(B(),1),t=V(H(),1);function ne(a){return a.toSorted((e,r)=>e.provider==="env"?1:r.provider==="env"?-1:0)}const le=a=>{let e=(0,oe.c)(43),{providerNames:r,onClose:A,onSuccess:F}=a,{writeSecret:Y}=X(),[o,M]=T.useState(""),[l,R]=T.useState(""),[n,z]=T.useState(r[0]),g;e[0]!==o||e[1]!==n||e[2]!==F||e[3]!==l||e[4]!==Y?(g=async i=>{if(i.preventDefault(),!n){E({title:"Error",description:"No location selected for the secret.",variant:"danger"});return}if(!o||!l||!n){E({title:"Error",description:"Please fill in all fields.",variant:"danger"});return}try{await Y({key:o,value:l,provider:"dotenv",name:n}),E({title:"Secret created",description:"The secret has been created successfully."}),F(o)}catch{E({title:"Error",description:"Failed to create secret. Please try again.",variant:"danger"})}},e[0]=o,e[1]=n,e[2]=F,e[3]=l,e[4]=Y,e[5]=g):g=e[5];let q=g,j;e[6]===Symbol.for("react.memo_cache_sentinel")?(j=(0,t.jsxs)(Z,{children:[(0,t.jsx)($,{children:"Add Secret"}),(0,t.jsx)(re,{children:"Add a new secret to your environment variables."})]}),e[6]=j):j=e[6];let y;e[7]===Symbol.for("react.memo_cache_sentinel")?(y=(0,t.jsx)(P,{htmlFor:"key",children:"Key"}),e[7]=y):y=e[7];let S;e[8]===Symbol.for("react.memo_cache_sentinel")?(S=i=>{M(ie(i.target.value))},e[8]=S):S=e[8];let s;e[9]===o?s=e[10]:(s=(0,t.jsxs)("div",{className:"grid gap-2",children:[y,(0,t.jsx)(D,{id:"key",value:o,onChange:S,placeholder:"MY_SECRET_KEY",required:!0})]}),e[9]=o,e[10]=s);let _;e[11]===Symbol.for("react.memo_cache_sentinel")?(_=(0,t.jsx)(P,{htmlFor:"value",children:"Value"}),e[11]=_):_=e[11];let b;e[12]===Symbol.for("react.memo_cache_sentinel")?(b=i=>R(i.target.value),e[12]=b):b=e[12];let c;e[13]===l?c=e[14]:(c=(0,t.jsx)(D,{id:"value",type:"password",value:l,onChange:b,required:!0,autoComplete:"off"}),e[13]=l,e[14]=c);let C;e[15]===Symbol.for("react.memo_cache_sentinel")?(C=se()&&(0,t.jsx)(L,{children:"Note: You are sending this key over http."}),e[15]=C):C=e[15];let d;e[16]===c?d=e[17]:(d=(0,t.jsxs)("div",{className:"grid gap-2",children:[_,c,C]}),e[16]=c,e[17]=d);let N;e[18]===Symbol.for("react.memo_cache_sentinel")?(N=(0,t.jsx)(P,{htmlFor:"location",children:"Location"}),e[18]=N):N=e[18];let m;e[19]===r.length?m=e[20]:(m=r.length===0&&(0,t.jsx)("p",{className:"text-sm text-muted-foreground",children:"No dotenv locations configured."}),e[19]=r.length,e[20]=m);let h;e[21]!==n||e[22]!==r?(h=r.length>0&&(0,t.jsxs)(U,{value:n,onValueChange:i=>z(i),children:[(0,t.jsx)(Q,{children:(0,t.jsx)(I,{placeholder:"Select a provider"})}),(0,t.jsx)(O,{children:r.map(ce)})]}),e[21]=n,e[22]=r,e[23]=h):h=e[23];let k;e[24]===Symbol.for("react.memo_cache_sentinel")?(k=(0,t.jsxs)(L,{children:["You can configure the location by setting the"," ",(0,t.jsx)(ae,{href:"https://links.marimo.app/dotenv",children:"dotenv configuration"}),"."]}),e[24]=k):k=e[24];let p;e[25]!==m||e[26]!==h?(p=(0,t.jsxs)("div",{className:"grid gap-2",children:[N,m,h,k]}),e[25]=m,e[26]=h,e[27]=p):p=e[27];let u;e[28]!==d||e[29]!==p||e[30]!==s?(u=(0,t.jsxs)("div",{className:"grid gap-4 py-4",children:[s,d,p]}),e[28]=d,e[29]=p,e[30]=s,e[31]=u):u=e[31];let f;e[32]===A?f=e[33]:(f=(0,t.jsx)(W,{type:"button",variant:"outline",onClick:A,children:"Cancel"}),e[32]=A,e[33]=f);let K=!o||!l||!n,v;e[34]===K?v=e[35]:(v=(0,t.jsx)(W,{type:"submit",disabled:K,children:"Add Secret"}),e[34]=K,e[35]=v);let x;e[36]!==f||e[37]!==v?(x=(0,t.jsxs)(ee,{children:[f,v]}),e[36]=f,e[37]=v,e[38]=x):x=e[38];let w;return e[39]!==q||e[40]!==u||e[41]!==x?(w=(0,t.jsx)(te,{children:(0,t.jsxs)("form",{onSubmit:q,children:[j,u,x]})}),e[39]=q,e[40]=u,e[41]=x,e[42]=w):w=e[42],w};function ie(a){return a.replaceAll(/\W/g,"_")}function se(){return window.location.href.startsWith("http://")}function ce(a){return(0,t.jsx)(J,{value:a,children:a},a)}export{ne as n,le as t};
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-CE1XkZU_.js"></script>
69
+ <script type="module" crossorigin src="./assets/index-BQp8477C.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">
@@ -91,7 +91,7 @@
91
91
  <link rel="modulepreload" crossorigin href="./assets/x-C-6liIBr.js">
92
92
  <link rel="modulepreload" crossorigin href="./assets/select-DZcFyKFQ.js">
93
93
  <link rel="modulepreload" crossorigin href="./assets/tooltip-DTV9tlSr.js">
94
- <link rel="modulepreload" crossorigin href="./assets/use-toast-IPAtJGea.js">
94
+ <link rel="modulepreload" crossorigin href="./assets/use-toast-CJ1Tj2eC.js">
95
95
  <link rel="modulepreload" crossorigin href="./assets/useEvent-D91BmmQi.js">
96
96
  <link rel="modulepreload" crossorigin href="./assets/invariant-BUdrueMv.js">
97
97
  <link rel="modulepreload" crossorigin href="./assets/isObject-DvLSfCY5.js">
@@ -133,10 +133,10 @@
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-BsaJrRWi.js">
136
+ <link rel="modulepreload" crossorigin href="./assets/cells-DrYr16TO.js">
137
137
  <link rel="modulepreload" crossorigin href="./assets/ErrorBoundary-DyYDV0HI.js">
138
138
  <link rel="modulepreload" crossorigin href="./assets/kbd-CTUAEnEx.js">
139
- <link rel="modulepreload" crossorigin href="./assets/useInstallPackage-DUF4IRRI.js">
139
+ <link rel="modulepreload" crossorigin href="./assets/useInstallPackage-GI6gu7_S.js">
140
140
  <link rel="modulepreload" crossorigin href="./assets/alert-dialog-BqFLkbUc.js">
141
141
  <link rel="modulepreload" crossorigin href="./assets/dialog-DzC_QCtT.js">
142
142
  <link rel="modulepreload" crossorigin href="./assets/useDebounce-7xZwVnnu.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-B1pHtGWp.js">
150
+ <link rel="modulepreload" crossorigin href="./assets/cell-link-CB8WeUi5.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-BiCBGyeW.js">
156
+ <link rel="modulepreload" crossorigin href="./assets/useRunCells-B7Ie4Pwb.js">
157
157
  <link rel="modulepreload" crossorigin href="./assets/copy-LK56fFow.js">
158
158
  <link rel="modulepreload" crossorigin href="./assets/copy-BwrPA9zQ.js">
159
- <link rel="modulepreload" crossorigin href="./assets/copy-icon-OjtDb4gO.js">
160
- <link rel="modulepreload" crossorigin href="./assets/RenderHTML-ChwPJgPd.js">
161
- <link rel="modulepreload" crossorigin href="./assets/datasource-DjDpJnbt.js">
162
- <link rel="modulepreload" crossorigin href="./assets/state-D-2KaAHK.js">
159
+ <link rel="modulepreload" crossorigin href="./assets/copy-icon-D_qz1txR.js">
160
+ <link rel="modulepreload" crossorigin href="./assets/RenderHTML-BlI-DkEC.js">
161
+ <link rel="modulepreload" crossorigin href="./assets/datasource-BKe0SLTo.js">
162
+ <link rel="modulepreload" crossorigin href="./assets/state-DttgoaV8.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-B0h5OVW9.js">
165
+ <link rel="modulepreload" crossorigin href="./assets/MarimoErrorOutput-D8_FYh4M.js">
166
166
  <link rel="modulepreload" crossorigin href="./assets/spinner-Bhir8k53.js">
167
- <link rel="modulepreload" crossorigin href="./assets/html-to-image-BWL3Ct4n.js">
168
- <link rel="modulepreload" crossorigin href="./assets/focus-q38vUCl-.js">
167
+ <link rel="modulepreload" crossorigin href="./assets/html-to-image-C9fYqzra.js">
168
+ <link rel="modulepreload" crossorigin href="./assets/focus-1NHETejS.js">
169
169
  <link rel="modulepreload" crossorigin href="./assets/useAsyncData-bgszE9F0.js">
170
- <link rel="modulepreload" crossorigin href="./assets/LazyAnyLanguageCodeMirror-BEvXb3VX.js">
170
+ <link rel="modulepreload" crossorigin href="./assets/LazyAnyLanguageCodeMirror-DXGWNDPu.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-CU5Lbyax.js">
173
+ <link rel="modulepreload" crossorigin href="./assets/markdown-renderer-DY7HNEoY.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-DT8gf0PL.js">
178
+ <link rel="modulepreload" crossorigin href="./assets/download-DZUw3TyK.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,38 +196,38 @@
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-BJaeXQim.js">
199
+ <link rel="modulepreload" crossorigin href="./assets/JsonOutput-CPYntVn7.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-BzBUxUz0.js">
207
- <link rel="modulepreload" crossorigin href="./assets/column-preview-BPismotx.js">
206
+ <link rel="modulepreload" crossorigin href="./assets/components-CB5nynlN.js">
207
+ <link rel="modulepreload" crossorigin href="./assets/column-preview-B16EcXyK.js">
208
208
  <link rel="modulepreload" crossorigin href="./assets/icons-Ol38nIbL.js">
209
- <link rel="modulepreload" crossorigin href="./assets/radio-group-DlVSuZ_1.js">
210
- <link rel="modulepreload" crossorigin href="./assets/floating-outline-BN9sAcQ0.js">
209
+ <link rel="modulepreload" crossorigin href="./assets/radio-group-2AaAWNrb.js">
210
+ <link rel="modulepreload" crossorigin href="./assets/floating-outline-3J8SCP2i.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-D3_kHsDx.js">
213
+ <link rel="modulepreload" crossorigin href="./assets/readonly-python-code-8smCAF83.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-BiMOJy0r.js">
217
+ <link rel="modulepreload" crossorigin href="./assets/file-icons-C8WXgGhY.js">
218
218
  <link rel="modulepreload" crossorigin href="./assets/switch-BiU_sAcn.js">
219
219
  <link rel="modulepreload" crossorigin href="./assets/events-CPoJAfgx.js">
220
220
  <link rel="modulepreload" crossorigin href="./assets/globals-7JnRMGd4.js">
221
- <link rel="modulepreload" crossorigin href="./assets/share-DMwTZOTH.js">
221
+ <link rel="modulepreload" crossorigin href="./assets/share-mSf5FZIP.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-BTTshiKq.js">
226
+ <link rel="modulepreload" crossorigin href="./assets/state-ZFlMDeHg.js">
227
227
  <link rel="modulepreload" crossorigin href="./assets/label-xHqFtfdz.js">
228
- <link rel="modulepreload" crossorigin href="./assets/textarea-DwltG8V6.js">
228
+ <link rel="modulepreload" crossorigin href="./assets/textarea-Cxjg9eKY.js">
229
229
  <link rel="modulepreload" crossorigin href="./assets/refresh-ccw-C-n2VFP5.js">
230
- <link rel="modulepreload" crossorigin href="./assets/form-BtBTvVVd.js">
230
+ <link rel="modulepreload" crossorigin href="./assets/form-66bEiwKr.js">
231
231
  <link rel="modulepreload" crossorigin href="./assets/renderShortcut-CcFk3m01.js">
232
232
  <link rel="modulepreload" crossorigin href="./assets/useBoolean-Bp18o6XG.js">
233
233
  <link rel="modulepreload" crossorigin href="./assets/useDeepCompareMemoize-zUHU--0D.js">
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@marimo-team/frontend",
3
- "version": "0.23.10-dev1",
3
+ "version": "0.23.10-dev13",
4
4
  "main": "dist/main.js",
5
5
  "types": "dist/index.d.ts",
6
6
  "type": "module",
@@ -57,7 +57,7 @@
57
57
  "@marimo-team/codemirror-ai": "^0.3.7",
58
58
  "@marimo-team/codemirror-languageserver": "^1.16.12",
59
59
  "@marimo-team/codemirror-mcp": "^0.1.5",
60
- "@marimo-team/codemirror-sql": "^0.2.4",
60
+ "@marimo-team/codemirror-sql": "^0.2.8",
61
61
  "@marimo-team/llm-info": "workspace:*",
62
62
  "@marimo-team/marimo-api": "workspace:*",
63
63
  "@marimo-team/react-slotz": "^0.2.0",
@@ -5,8 +5,8 @@ import type {
5
5
  RowSelectionState,
6
6
  SortingState,
7
7
  } from "@tanstack/react-table";
8
- import { fireEvent, render, screen, within } from "@testing-library/react";
9
- import { describe, expect, it, vi } from "vitest";
8
+ import { act, fireEvent, render, screen, within } from "@testing-library/react";
9
+ import { afterEach, describe, expect, it, vi } from "vitest";
10
10
  import { TooltipProvider } from "@/components/ui/tooltip";
11
11
  import { DataTable } from "../data-table";
12
12
 
@@ -16,6 +16,12 @@ interface TestData {
16
16
  }
17
17
 
18
18
  describe("DataTable", () => {
19
+ // Restore real timers unconditionally so a failed assertion in a
20
+ // fake-timer test can't leak fake timers into later tests.
21
+ afterEach(() => {
22
+ vi.useRealTimers();
23
+ });
24
+
19
25
  it("should maintain selection state when remounted", () => {
20
26
  const mockOnRowSelectionChange = vi.fn();
21
27
  const testData: TestData[] = [
@@ -63,17 +69,15 @@ describe("DataTable", () => {
63
69
  expect(commonProps.rowSelection).toEqual(initialRowSelection);
64
70
  });
65
71
 
66
- it("applies hoverTemplate to the row title using row values", () => {
72
+ it("shows the hoverTemplate text as a styled tooltip on hover", async () => {
73
+ vi.useFakeTimers();
67
74
  interface RowData {
68
75
  id: number;
69
76
  first: string;
70
77
  last: string;
71
78
  }
72
79
 
73
- const testData: RowData[] = [
74
- { id: 1, first: "Michael", last: "Scott" },
75
- { id: 2, first: "Jim", last: "Halpert" },
76
- ];
80
+ const testData: RowData[] = [{ id: 1, first: "Michael", last: "Scott" }];
77
81
 
78
82
  const columns: ColumnDef<RowData>[] = [
79
83
  { accessorKey: "first", header: "First" },
@@ -86,7 +90,7 @@ describe("DataTable", () => {
86
90
  data={testData}
87
91
  columns={columns}
88
92
  selection={null}
89
- totalRows={2}
93
+ totalRows={1}
90
94
  totalColumns={2}
91
95
  pagination={false}
92
96
  hoverTemplate={"{{first}} {{last}}"}
@@ -94,11 +98,149 @@ describe("DataTable", () => {
94
98
  </TooltipProvider>,
95
99
  );
96
100
 
97
- // Grab all rows and assert title attribute computed from template
98
101
  const rows = screen.getAllByRole("row");
99
- // The first row is header; subsequent rows correspond to data
100
- expect(rows[1]).toHaveAttribute("title", "Michael Scott");
101
- expect(rows[2]).toHaveAttribute("title", "Jim Halpert");
102
+ // Native title is gone; hover text now comes from the styled tooltip.
103
+ expect(rows[1]).not.toHaveAttribute("title");
104
+
105
+ const cell = within(rows[1]).getAllByRole("cell")[0];
106
+ fireEvent.mouseOver(cell, { buttons: 0 });
107
+ act(() => {
108
+ vi.advanceTimersByTime(400);
109
+ });
110
+
111
+ // Radix renders the content twice (visible + a11y-hidden), so match all.
112
+ expect(screen.getAllByText("Michael Scott").length).toBeGreaterThan(0);
113
+ });
114
+
115
+ it("shows per-cell hover text from cellHoverTexts on hover", () => {
116
+ vi.useFakeTimers();
117
+ const testData: TestData[] = [{ id: 1, name: "Test 1" }];
118
+ const columns: ColumnDef<TestData>[] = [
119
+ { id: "name", accessorKey: "name", header: "Name" },
120
+ ];
121
+
122
+ render(
123
+ <TooltipProvider>
124
+ <DataTable
125
+ data={testData}
126
+ columns={columns}
127
+ selection={null}
128
+ totalRows={1}
129
+ totalColumns={1}
130
+ pagination={false}
131
+ cellHoverTexts={{ "0": { name: "per-cell tip" } }}
132
+ />
133
+ </TooltipProvider>,
134
+ );
135
+
136
+ const cell = within(screen.getAllByRole("row")[1]).getByRole("cell");
137
+ fireEvent.mouseOver(cell, { buttons: 0 });
138
+ act(() => {
139
+ vi.advanceTimersByTime(400);
140
+ });
141
+
142
+ expect(screen.getAllByText("per-cell tip").length).toBeGreaterThan(0);
143
+ });
144
+
145
+ it("links the focused cell to the tooltip content for assistive tech", () => {
146
+ const testData: TestData[] = [{ id: 1, name: "Test 1" }];
147
+ const columns: ColumnDef<TestData>[] = [
148
+ { id: "name", accessorKey: "name", header: "Name" },
149
+ ];
150
+
151
+ render(
152
+ <TooltipProvider>
153
+ <DataTable
154
+ data={testData}
155
+ columns={columns}
156
+ selection={null}
157
+ totalRows={1}
158
+ totalColumns={1}
159
+ pagination={false}
160
+ cellHoverTexts={{ "0": { name: "focus tip" } }}
161
+ />
162
+ </TooltipProvider>,
163
+ );
164
+
165
+ const cell = within(screen.getAllByRole("row")[1]).getByRole("cell");
166
+ fireEvent.focus(cell);
167
+
168
+ const describedBy = cell.getAttribute("aria-describedby");
169
+ expect(describedBy).toBeTruthy();
170
+ expect(document.getElementById(describedBy as string)).toHaveTextContent(
171
+ "focus tip",
172
+ );
173
+
174
+ fireEvent.blur(cell);
175
+ expect(cell).not.toHaveAttribute("aria-describedby");
176
+ });
177
+
178
+ it("does not show a tooltip on pointer-induced focus", () => {
179
+ const testData: TestData[] = [{ id: 1, name: "Test 1" }];
180
+ const columns: ColumnDef<TestData>[] = [
181
+ { id: "name", accessorKey: "name", header: "Name" },
182
+ ];
183
+
184
+ render(
185
+ <TooltipProvider>
186
+ <DataTable
187
+ data={testData}
188
+ columns={columns}
189
+ selection={null}
190
+ totalRows={1}
191
+ totalColumns={1}
192
+ pagination={false}
193
+ cellHoverTexts={{ "0": { name: "click tip" } }}
194
+ />
195
+ </TooltipProvider>,
196
+ );
197
+
198
+ const cell = within(screen.getAllByRole("row")[1]).getByRole("cell");
199
+ // A click focuses the cell; the resulting focus must not show a tooltip.
200
+ fireEvent.mouseDown(cell);
201
+ fireEvent.focus(cell);
202
+
203
+ expect(cell).not.toHaveAttribute("aria-describedby");
204
+ expect(screen.queryByText("click tip")).toBeNull();
205
+ });
206
+
207
+ it("does not let a pending hover timer overwrite a focus tooltip", () => {
208
+ vi.useFakeTimers();
209
+ interface RowData {
210
+ id: number;
211
+ a: string;
212
+ b: string;
213
+ }
214
+ const testData: RowData[] = [{ id: 1, a: "a", b: "b" }];
215
+ const columns: ColumnDef<RowData>[] = [
216
+ { id: "a", accessorKey: "a", header: "A" },
217
+ { id: "b", accessorKey: "b", header: "B" },
218
+ ];
219
+
220
+ render(
221
+ <TooltipProvider>
222
+ <DataTable
223
+ data={testData}
224
+ columns={columns}
225
+ selection={null}
226
+ totalRows={1}
227
+ totalColumns={2}
228
+ pagination={false}
229
+ cellHoverTexts={{ "0": { a: "hover A", b: "focus B" } }}
230
+ />
231
+ </TooltipProvider>,
232
+ );
233
+
234
+ const cells = within(screen.getAllByRole("row")[1]).getAllByRole("cell");
235
+ // Start a pending hover-show on cell A, then focus cell B before it fires.
236
+ fireEvent.mouseOver(cells[0], { buttons: 0 });
237
+ fireEvent.focus(cells[1]);
238
+ act(() => {
239
+ vi.advanceTimersByTime(400);
240
+ });
241
+
242
+ expect(screen.getAllByText("focus B").length).toBeGreaterThan(0);
243
+ expect(screen.queryByText("hover A")).toBeNull();
102
244
  });
103
245
 
104
246
  it("does not virtualize small datasets without pagination", () => {
@@ -0,0 +1,60 @@
1
+ /* Copyright 2026 Marimo. All rights reserved. */
2
+ import type { Cell } from "@tanstack/react-table";
3
+ import { describe, expect, it } from "vitest";
4
+ import { applyHoverTemplate, computeCellTooltipContent } from "../content";
5
+
6
+ function fakeCell(columnId: string, value: unknown, hoverTitle?: string) {
7
+ return {
8
+ column: { id: columnId },
9
+ getValue: () => value,
10
+ getHoverTitle: () => hoverTitle,
11
+ } as unknown as Cell<unknown, unknown>;
12
+ }
13
+
14
+ describe("applyHoverTemplate", () => {
15
+ it("substitutes column placeholders", () => {
16
+ const cells = [fakeCell("first", "Michael"), fakeCell("last", "Scott")];
17
+ expect(applyHoverTemplate("{{first}} {{last}}", cells)).toBe(
18
+ "Michael Scott",
19
+ );
20
+ });
21
+
22
+ it("renders nulls as empty strings", () => {
23
+ const cells = [fakeCell("a", null)];
24
+ expect(applyHoverTemplate("[{{a}}]", cells)).toBe("[]");
25
+ });
26
+
27
+ it("leaves unknown placeholders intact", () => {
28
+ expect(applyHoverTemplate("{{missing}}", [fakeCell("a", 1)])).toBe(
29
+ "{{missing}}",
30
+ );
31
+ });
32
+ });
33
+
34
+ describe("computeCellTooltipContent", () => {
35
+ it("prefers cell-level hover title", () => {
36
+ const cell = fakeCell("a", 1, "cell text");
37
+ expect(computeCellTooltipContent(cell, "{{a}}")).toBe("cell text");
38
+ });
39
+
40
+ it("falls back to row template when no cell title", () => {
41
+ const cell = {
42
+ column: { id: "first" },
43
+ getValue: () => "X",
44
+ getHoverTitle: () => undefined,
45
+ row: {
46
+ getVisibleCells: () => [
47
+ fakeCell("first", "Jim"),
48
+ fakeCell("last", "Halpert"),
49
+ ],
50
+ },
51
+ } as unknown as Cell<unknown, unknown>;
52
+ expect(computeCellTooltipContent(cell, "{{first}} {{last}}")).toBe(
53
+ "Jim Halpert",
54
+ );
55
+ });
56
+
57
+ it("returns undefined with no title and no template", () => {
58
+ expect(computeCellTooltipContent(fakeCell("a", 1), null)).toBeUndefined();
59
+ });
60
+ });
@@ -0,0 +1,44 @@
1
+ /* Copyright 2026 Marimo. All rights reserved. */
2
+ import type { Cell, RowData } from "@tanstack/react-table";
3
+ import type { ReactNode } from "react";
4
+ import { stringifyUnknownValue } from "../utils";
5
+
6
+ export function applyHoverTemplate<TData extends RowData>(
7
+ template: string,
8
+ cells: Cell<TData, unknown>[],
9
+ ): string {
10
+ const variableRegex = /{{(\w+)}}/g;
11
+ const idToValue = new Map<string, string>();
12
+ for (const c of cells) {
13
+ const s = stringifyUnknownValue({
14
+ value: c.getValue(),
15
+ nullAsEmptyString: true,
16
+ });
17
+ idToValue.set(c.column.id, s);
18
+ }
19
+ return template.replaceAll(variableRegex, (_substr, varName: string) => {
20
+ const val = idToValue.get(varName);
21
+ return val === undefined ? `{{${varName}}}` : val;
22
+ });
23
+ }
24
+
25
+ /**
26
+ * Resolve the tooltip content for a hovered cell.
27
+ *
28
+ * Cell-level (callable `hover_template`) takes precedence; otherwise the
29
+ * row-level string template is rendered against the row's visible cells.
30
+ * Returns `undefined` when there is nothing to show.
31
+ */
32
+ export function computeCellTooltipContent<TData extends RowData>(
33
+ cell: Cell<TData, unknown>,
34
+ hoverTemplate: string | null,
35
+ ): ReactNode {
36
+ const cellTitle = cell.getHoverTitle?.();
37
+ if (cellTitle != null && cellTitle !== "") {
38
+ return cellTitle;
39
+ }
40
+ if (hoverTemplate) {
41
+ return applyHoverTemplate(hoverTemplate, cell.row.getVisibleCells());
42
+ }
43
+ return undefined;
44
+ }
@@ -0,0 +1,55 @@
1
+ /* Copyright 2026 Marimo. All rights reserved. */
2
+ import {
3
+ TooltipContent,
4
+ TooltipPortal,
5
+ TooltipRoot,
6
+ TooltipTrigger,
7
+ } from "@/components/ui/tooltip";
8
+ import type { HoverTooltipState } from "./use-table-hover-tooltip";
9
+
10
+ interface HoverTooltipProps {
11
+ state: HoverTooltipState | null;
12
+ contentId: string;
13
+ onClose: () => void;
14
+ }
15
+
16
+ /**
17
+ * A single radix tooltip whose anchor is repositioned to the hovered cell.
18
+ * Rendering one instance per table (instead of one per cell) keeps the cost
19
+ * constant regardless of how many cells are on screen.
20
+ */
21
+ export const HoverTooltip = ({
22
+ state,
23
+ contentId,
24
+ onClose,
25
+ }: HoverTooltipProps) => {
26
+ return (
27
+ <TooltipRoot
28
+ open={state != null}
29
+ onOpenChange={(open) => {
30
+ if (!open) {
31
+ onClose();
32
+ }
33
+ }}
34
+ delayDuration={0}
35
+ disableHoverableContent={true}
36
+ >
37
+ <TooltipTrigger asChild={true}>
38
+ <div
39
+ aria-hidden={true}
40
+ style={{
41
+ position: "fixed",
42
+ top: state?.rect.top ?? 0,
43
+ left: state?.rect.left ?? 0,
44
+ width: state?.rect.width ?? 0,
45
+ height: state?.rect.height ?? 0,
46
+ pointerEvents: "none",
47
+ }}
48
+ />
49
+ </TooltipTrigger>
50
+ <TooltipPortal>
51
+ <TooltipContent id={contentId}>{state?.content}</TooltipContent>
52
+ </TooltipPortal>
53
+ </TooltipRoot>
54
+ );
55
+ };