drizzle-cube 0.3.13 → 0.3.15

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 (182) hide show
  1. package/dist/adapters/{compiler-DTOU8IsK.js → compiler-DdcGVWIl.js} +7429 -4165
  2. package/dist/adapters/compiler-mAJDpIQx.cjs +214 -0
  3. package/dist/adapters/express/index.cjs +1 -1
  4. package/dist/adapters/express/index.d.ts +1 -1
  5. package/dist/adapters/express/index.js +1 -1
  6. package/dist/adapters/fastify/index.cjs +1 -1
  7. package/dist/adapters/fastify/index.d.ts +1 -1
  8. package/dist/adapters/fastify/index.js +1 -1
  9. package/dist/adapters/hono/index.cjs +1 -1
  10. package/dist/adapters/hono/index.d.ts +1 -1
  11. package/dist/adapters/hono/index.js +1 -1
  12. package/dist/adapters/nextjs/index.cjs +1 -1
  13. package/dist/adapters/nextjs/index.d.ts +1 -1
  14. package/dist/adapters/nextjs/index.js +1 -1
  15. package/dist/adapters/utils.d.ts +29 -0
  16. package/dist/client/adapters/index.d.ts +2 -0
  17. package/dist/client/adapters/retentionModeAdapter.d.ts +3 -0
  18. package/dist/client/charts.js +12 -12
  19. package/dist/client/chunks/RetentionCombinedChart-CqMAkdcR.js +256 -0
  20. package/dist/client/chunks/RetentionCombinedChart-CqMAkdcR.js.map +1 -0
  21. package/dist/client/chunks/RetentionCombinedChart.config-C-ILIaEb.js +47 -0
  22. package/dist/client/chunks/RetentionCombinedChart.config-C-ILIaEb.js.map +1 -0
  23. package/dist/client/chunks/RetentionHeatmap-Bofadstm.js +178 -0
  24. package/dist/client/chunks/RetentionHeatmap-Bofadstm.js.map +1 -0
  25. package/dist/client/chunks/RetentionHeatmap.config-CIvhc-GT.js +28 -0
  26. package/dist/client/chunks/RetentionHeatmap.config-CIvhc-GT.js.map +1 -0
  27. package/dist/client/chunks/analysis-builder-D0MIKiKS.js +6939 -0
  28. package/dist/client/chunks/analysis-builder-D0MIKiKS.js.map +1 -0
  29. package/dist/client/chunks/analysis-builder-shared-gS0TDC6Y.js +2779 -0
  30. package/dist/client/chunks/analysis-builder-shared-gS0TDC6Y.js.map +1 -0
  31. package/dist/client/chunks/{chart-activity-grid-CuPARsr1.js → chart-activity-grid-B37W8fcE.js} +11 -11
  32. package/dist/client/chunks/{chart-activity-grid-CuPARsr1.js.map → chart-activity-grid-B37W8fcE.js.map} +1 -1
  33. package/dist/client/chunks/{chart-area-cPrJnhLj.js → chart-area-DZcxSVB1.js} +2 -2
  34. package/dist/client/chunks/{chart-area-cPrJnhLj.js.map → chart-area-DZcxSVB1.js.map} +1 -1
  35. package/dist/client/chunks/{chart-bar-D68HFPpa.js → chart-bar-BFAdRj-E.js} +2 -2
  36. package/dist/client/chunks/{chart-bar-D68HFPpa.js.map → chart-bar-BFAdRj-E.js.map} +1 -1
  37. package/dist/client/chunks/{chart-bubble-CquyYfNO.js → chart-bubble-eoCCbl3h.js} +2 -2
  38. package/dist/client/chunks/{chart-bubble-CquyYfNO.js.map → chart-bubble-eoCCbl3h.js.map} +1 -1
  39. package/dist/client/chunks/{chart-config-activity-grid-Bkvx0F-G.js → chart-config-activity-grid-BBSNCbkb.js} +2 -2
  40. package/dist/client/chunks/{chart-config-activity-grid-Bkvx0F-G.js.map → chart-config-activity-grid-BBSNCbkb.js.map} +1 -1
  41. package/dist/client/chunks/{chart-config-area-OApsRaYC.js → chart-config-area-CyyJOO2T.js} +2 -2
  42. package/dist/client/chunks/{chart-config-area-OApsRaYC.js.map → chart-config-area-CyyJOO2T.js.map} +1 -1
  43. package/dist/client/chunks/{chart-config-bar-Dy21oaIA.js → chart-config-bar-BJKGnfLt.js} +2 -2
  44. package/dist/client/chunks/{chart-config-bar-Dy21oaIA.js.map → chart-config-bar-BJKGnfLt.js.map} +1 -1
  45. package/dist/client/chunks/{chart-config-bubble-Chv0SoFm.js → chart-config-bubble-CuSsCHZ4.js} +2 -2
  46. package/dist/client/chunks/{chart-config-bubble-Chv0SoFm.js.map → chart-config-bubble-CuSsCHZ4.js.map} +1 -1
  47. package/dist/client/chunks/{chart-config-data-table-DTIdC35a.js → chart-config-data-table-BhgqwoqT.js} +2 -2
  48. package/dist/client/chunks/{chart-config-data-table-DTIdC35a.js.map → chart-config-data-table-BhgqwoqT.js.map} +1 -1
  49. package/dist/client/chunks/{chart-config-funnel-f17Livgr.js → chart-config-funnel-BlSQYng0.js} +4 -4
  50. package/dist/client/chunks/{chart-config-funnel-f17Livgr.js.map → chart-config-funnel-BlSQYng0.js.map} +1 -1
  51. package/dist/client/chunks/{chart-config-heat-map-DPhNICha.js → chart-config-heat-map-DHQGFZhX.js} +2 -2
  52. package/dist/client/chunks/{chart-config-heat-map-DPhNICha.js.map → chart-config-heat-map-DHQGFZhX.js.map} +1 -1
  53. package/dist/client/chunks/{chart-config-kpi-delta-CCl1d-St.js → chart-config-kpi-delta-yTA5ug_l.js} +2 -2
  54. package/dist/client/chunks/{chart-config-kpi-delta-CCl1d-St.js.map → chart-config-kpi-delta-yTA5ug_l.js.map} +1 -1
  55. package/dist/client/chunks/{chart-config-kpi-number-DkE3eSwH.js → chart-config-kpi-number-nVAwDXzq.js} +2 -2
  56. package/dist/client/chunks/{chart-config-kpi-number-DkE3eSwH.js.map → chart-config-kpi-number-nVAwDXzq.js.map} +1 -1
  57. package/dist/client/chunks/{chart-config-kpi-text-BMbgdxZm.js → chart-config-kpi-text-DZjqsx-b.js} +2 -2
  58. package/dist/client/chunks/{chart-config-kpi-text-BMbgdxZm.js.map → chart-config-kpi-text-DZjqsx-b.js.map} +1 -1
  59. package/dist/client/chunks/{chart-config-line-BnLlRUQE.js → chart-config-line-DR0ThxZy.js} +2 -2
  60. package/dist/client/chunks/{chart-config-line-BnLlRUQE.js.map → chart-config-line-DR0ThxZy.js.map} +1 -1
  61. package/dist/client/chunks/{chart-config-markdown-DIaMFC0Z.js → chart-config-markdown-DZxdGNVQ.js} +2 -2
  62. package/dist/client/chunks/{chart-config-markdown-DIaMFC0Z.js.map → chart-config-markdown-DZxdGNVQ.js.map} +1 -1
  63. package/dist/client/chunks/{chart-config-pie-O9y_T0BQ.js → chart-config-pie-BM5lgH-w.js} +2 -2
  64. package/dist/client/chunks/{chart-config-pie-O9y_T0BQ.js.map → chart-config-pie-BM5lgH-w.js.map} +1 -1
  65. package/dist/client/chunks/{chart-config-radar-CXa0354h.js → chart-config-radar-BBAVIF0S.js} +2 -2
  66. package/dist/client/chunks/{chart-config-radar-CXa0354h.js.map → chart-config-radar-BBAVIF0S.js.map} +1 -1
  67. package/dist/client/chunks/{chart-config-radial-bar-BppJU8-Q.js → chart-config-radial-bar-CTwjDRnB.js} +2 -2
  68. package/dist/client/chunks/{chart-config-radial-bar-BppJU8-Q.js.map → chart-config-radial-bar-CTwjDRnB.js.map} +1 -1
  69. package/dist/client/chunks/{chart-config-sankey-BfBHgL4x.js → chart-config-sankey-CNAgsMQ4.js} +2 -2
  70. package/dist/client/chunks/{chart-config-sankey-BfBHgL4x.js.map → chart-config-sankey-CNAgsMQ4.js.map} +1 -1
  71. package/dist/client/chunks/{chart-config-scatter-BTt8a10R.js → chart-config-scatter-CWvN2E-X.js} +2 -2
  72. package/dist/client/chunks/{chart-config-scatter-BTt8a10R.js.map → chart-config-scatter-CWvN2E-X.js.map} +1 -1
  73. package/dist/client/chunks/{chart-config-sunburst-DNmQpIIx.js → chart-config-sunburst-W_SKwaj0.js} +4 -4
  74. package/dist/client/chunks/{chart-config-sunburst-DNmQpIIx.js.map → chart-config-sunburst-W_SKwaj0.js.map} +1 -1
  75. package/dist/client/chunks/{chart-config-tree-map-HVgG4oa0.js → chart-config-tree-map-CLmRvvMR.js} +2 -2
  76. package/dist/client/chunks/{chart-config-tree-map-HVgG4oa0.js.map → chart-config-tree-map-CLmRvvMR.js.map} +1 -1
  77. package/dist/client/chunks/{chart-data-table-D4WDqbM0.js → chart-data-table-kudRwZxJ.js} +4 -4
  78. package/dist/client/chunks/{chart-data-table-D4WDqbM0.js.map → chart-data-table-kudRwZxJ.js.map} +1 -1
  79. package/dist/client/chunks/{chart-funnel-Csdn4FbN.js → chart-funnel-CA0XJkfh.js} +2 -2
  80. package/dist/client/chunks/{chart-funnel-Csdn4FbN.js.map → chart-funnel-CA0XJkfh.js.map} +1 -1
  81. package/dist/client/chunks/{chart-heat-map-v1afxnjq.js → chart-heat-map-DUy9_pWM.js} +2 -2
  82. package/dist/client/chunks/{chart-heat-map-v1afxnjq.js.map → chart-heat-map-DUy9_pWM.js.map} +1 -1
  83. package/dist/client/chunks/chart-kpi-delta-MIGGCpkG.js +351 -0
  84. package/dist/client/chunks/chart-kpi-delta-MIGGCpkG.js.map +1 -0
  85. package/dist/client/chunks/chart-kpi-number-DM0Brd91.js +473 -0
  86. package/dist/client/chunks/chart-kpi-number-DM0Brd91.js.map +1 -0
  87. package/dist/client/chunks/{chart-kpi-text-CRp8QWYG.js → chart-kpi-text-BAb28V4X.js} +3 -3
  88. package/dist/client/chunks/{chart-kpi-text-CRp8QWYG.js.map → chart-kpi-text-BAb28V4X.js.map} +1 -1
  89. package/dist/client/chunks/{chart-line-DqqE7ky9.js → chart-line-e3h8sa1R.js} +5 -5
  90. package/dist/client/chunks/{chart-line-DqqE7ky9.js.map → chart-line-e3h8sa1R.js.map} +1 -1
  91. package/dist/client/chunks/{chart-pie-B5WBzIRH.js → chart-pie-CFoHYqDB.js} +2 -2
  92. package/dist/client/chunks/{chart-pie-B5WBzIRH.js.map → chart-pie-CFoHYqDB.js.map} +1 -1
  93. package/dist/client/chunks/{chart-radar-DL_dvhA-.js → chart-radar-CPPwkfxj.js} +2 -2
  94. package/dist/client/chunks/{chart-radar-DL_dvhA-.js.map → chart-radar-CPPwkfxj.js.map} +1 -1
  95. package/dist/client/chunks/{chart-radial-bar-DDRo6nz-.js → chart-radial-bar-BdEOM-P1.js} +2 -2
  96. package/dist/client/chunks/{chart-radial-bar-DDRo6nz-.js.map → chart-radial-bar-BdEOM-P1.js.map} +1 -1
  97. package/dist/client/chunks/{chart-sankey-C_bgIfg-.js → chart-sankey-Bbady-8g.js} +2 -2
  98. package/dist/client/chunks/{chart-sankey-C_bgIfg-.js.map → chart-sankey-Bbady-8g.js.map} +1 -1
  99. package/dist/client/chunks/{chart-scatter-DjmJRlK0.js → chart-scatter-JFu0Pv3a.js} +15 -15
  100. package/dist/client/chunks/{chart-scatter-DjmJRlK0.js.map → chart-scatter-JFu0Pv3a.js.map} +1 -1
  101. package/dist/client/chunks/{chart-sunburst-CbMEnaes.js → chart-sunburst-sayxze15.js} +2 -2
  102. package/dist/client/chunks/{chart-sunburst-CbMEnaes.js.map → chart-sunburst-sayxze15.js.map} +1 -1
  103. package/dist/client/chunks/{chart-tree-map-DEfJtJVC.js → chart-tree-map-Ks2xev8b.js} +30 -30
  104. package/dist/client/chunks/{chart-tree-map-DEfJtJVC.js.map → chart-tree-map-Ks2xev8b.js.map} +1 -1
  105. package/dist/client/chunks/chartConfigRegistry-BumUIPw4.js +44 -0
  106. package/dist/client/chunks/{chartConfigRegistry-CiOq-PqX.js.map → chartConfigRegistry-BumUIPw4.js.map} +1 -1
  107. package/dist/client/chunks/charts-_yZ9gBJU.js +230 -0
  108. package/dist/client/chunks/charts-_yZ9gBJU.js.map +1 -0
  109. package/dist/client/chunks/{charts-core-CXrhEEVF.js → charts-core-Bzu9PzMd.js} +10 -10
  110. package/dist/client/chunks/{charts-core-CXrhEEVF.js.map → charts-core-Bzu9PzMd.js.map} +1 -1
  111. package/dist/client/chunks/{charts-loader-BtsnUO4Q.js → charts-loader-cMtx4zHx.js} +30 -28
  112. package/dist/client/chunks/charts-loader-cMtx4zHx.js.map +1 -0
  113. package/dist/client/chunks/{components-BDrlf9Er.js → components-DQuPThfA.js} +3575 -3208
  114. package/dist/client/chunks/components-DQuPThfA.js.map +1 -0
  115. package/dist/client/chunks/{core-B8zw0qRf.js → core-DwOXVb87.js} +2 -2
  116. package/dist/client/chunks/{core-B8zw0qRf.js.map → core-DwOXVb87.js.map} +1 -1
  117. package/dist/client/chunks/hooks-DPRv1Xhb.js +150 -0
  118. package/dist/client/chunks/{hooks-B8Zw5PfL.js.map → hooks-DPRv1Xhb.js.map} +1 -1
  119. package/dist/client/chunks/{icons-NzFHtqeM.js → icons-DRreo6m8.js} +128 -112
  120. package/dist/client/chunks/{icons-NzFHtqeM.js.map → icons-DRreo6m8.js.map} +1 -1
  121. package/dist/client/chunks/{providers-CqCiJTEj.js → providers-BW8D7Wso.js} +2 -2
  122. package/dist/client/chunks/{providers-CqCiJTEj.js.map → providers-BW8D7Wso.js.map} +1 -1
  123. package/dist/client/chunks/retention-CzCo8262.js +120 -0
  124. package/dist/client/chunks/retention-CzCo8262.js.map +1 -0
  125. package/dist/client/chunks/{useDirtyStateTracking-C_mitVwh.js → useDirtyStateTracking-Yu_qQXb-.js} +101 -99
  126. package/dist/client/chunks/useDirtyStateTracking-Yu_qQXb-.js.map +1 -0
  127. package/dist/client/chunks/{vendor-DzzxS7Ay.js → vendor-BSkQZgtm.js} +548 -540
  128. package/dist/client/chunks/vendor-BSkQZgtm.js.map +1 -0
  129. package/dist/client/components/AnalysisBuilder/AnalysisTypeSelector.d.ts +3 -1
  130. package/dist/client/components/AnalysisBuilder/RetentionConfigPanel.d.ts +36 -0
  131. package/dist/client/components/AnalysisBuilder/RetentionModeContent.d.ts +71 -0
  132. package/dist/client/components/AnalysisBuilder/types.d.ts +99 -0
  133. package/dist/client/components/DebugModal.d.ts +2 -1
  134. package/dist/client/components/charts/RetentionCombinedChart.config.d.ts +2 -0
  135. package/dist/client/components/charts/RetentionCombinedChart.d.ts +14 -0
  136. package/dist/client/components/charts/RetentionHeatmap.config.d.ts +2 -0
  137. package/dist/client/components/charts/RetentionHeatmap.d.ts +7 -0
  138. package/dist/client/components.js +2 -2
  139. package/dist/client/hooks/queries/index.d.ts +2 -1
  140. package/dist/client/hooks/queries/useDryRunQuery.d.ts +26 -0
  141. package/dist/client/hooks/queries/useExplainQuery.d.ts +3 -1
  142. package/dist/client/hooks/queries/useFlowQuery.d.ts +5 -0
  143. package/dist/client/hooks/queries/useRetentionQuery.d.ts +67 -0
  144. package/dist/client/hooks/useAnalysisBuilderHook.d.ts +61 -0
  145. package/dist/client/hooks/useAnalysisQueryExecution.d.ts +42 -1
  146. package/dist/client/hooks.js +3 -3
  147. package/dist/client/icons/customIcons.d.ts +7 -0
  148. package/dist/client/icons/types.d.ts +1 -0
  149. package/dist/client/icons.js +1 -1
  150. package/dist/client/index.js +11 -11
  151. package/dist/client/providers.js +1 -1
  152. package/dist/client/stores/analysisBuilderStore.d.ts +86 -1
  153. package/dist/client/stores/dashboardStore.d.ts +2 -1
  154. package/dist/client/stores/slices/index.d.ts +2 -0
  155. package/dist/client/stores/slices/retentionSlice.d.ts +66 -0
  156. package/dist/client/styles.css +1 -1
  157. package/dist/client/types/analysisConfig.d.ts +29 -4
  158. package/dist/client/types/funnel.d.ts +5 -0
  159. package/dist/client/types/retention.d.ts +301 -0
  160. package/dist/client/types.d.ts +6 -3
  161. package/dist/client-bundle-stats.html +1 -1
  162. package/dist/server/index.cjs +76 -54
  163. package/dist/server/index.d.ts +217 -0
  164. package/dist/server/index.js +7363 -4152
  165. package/package.json +5 -2
  166. package/dist/adapters/compiler-CO13DaEb.cjs +0 -192
  167. package/dist/client/chunks/analysis-builder-Dc9NrG_N.js +0 -6013
  168. package/dist/client/chunks/analysis-builder-Dc9NrG_N.js.map +0 -1
  169. package/dist/client/chunks/analysis-builder-shared-B3-UWqQ2.js +0 -2540
  170. package/dist/client/chunks/analysis-builder-shared-B3-UWqQ2.js.map +0 -1
  171. package/dist/client/chunks/chart-kpi-delta-Bs5R5xr4.js +0 -435
  172. package/dist/client/chunks/chart-kpi-delta-Bs5R5xr4.js.map +0 -1
  173. package/dist/client/chunks/chart-kpi-number-Cf4Pgkm9.js +0 -392
  174. package/dist/client/chunks/chart-kpi-number-Cf4Pgkm9.js.map +0 -1
  175. package/dist/client/chunks/chartConfigRegistry-CiOq-PqX.js +0 -44
  176. package/dist/client/chunks/charts-loader-BtsnUO4Q.js.map +0 -1
  177. package/dist/client/chunks/charts-xNubY0vm.js +0 -226
  178. package/dist/client/chunks/charts-xNubY0vm.js.map +0 -1
  179. package/dist/client/chunks/components-BDrlf9Er.js.map +0 -1
  180. package/dist/client/chunks/hooks-B8Zw5PfL.js +0 -123
  181. package/dist/client/chunks/useDirtyStateTracking-C_mitVwh.js.map +0 -1
  182. package/dist/client/chunks/vendor-DzzxS7Ay.js.map +0 -1
@@ -1,6 +1,6 @@
1
1
  import { useContext as C, createContext as E, useMemo as f, useState as x, useEffect as M, useCallback as g } from "react";
2
2
  import { jsx as p } from "react/jsx-runtime";
3
- import { u as z, a as D, Q as L, b as k } from "./vendor-DzzxS7Ay.js";
3
+ import { u as z, a as D, Q as L, b as k } from "./vendor-BSkQZgtm.js";
4
4
  class Q {
5
5
  apiUrl;
6
6
  headers;
@@ -545,4 +545,4 @@ export {
545
545
  oe as j,
546
546
  Z as u
547
547
  };
548
- //# sourceMappingURL=providers-CqCiJTEj.js.map
548
+ //# sourceMappingURL=providers-BW8D7Wso.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"providers-CqCiJTEj.js","sources":["../../../src/client/client/CubeClient.ts","../../../src/client/client/BatchCoordinator.ts","../../../src/client/providers/CubeApiProvider.tsx","../../../src/client/hooks/queries/useCubeMetaQuery.ts","../../../src/client/utils/thumbnail.ts","../../../src/client/providers/CubeFeaturesProvider.tsx","../../../src/client/providers/ScrollContainerContext.tsx","../../../src/client/providers/CubeMetaContext.tsx","../../../src/client/providers/CubeMetaProvider.tsx","../../../src/client/providers/CubeProvider.tsx"],"sourcesContent":["/**\n * Minimal Cube client implementation\n * Replaces @cubejs-client/core with lighter implementation\n */\n\nimport type { CubeQuery, CubeApiOptions, CubeResultSet, ExplainResult, ExplainOptions } from '../types'\n\nexport class CubeClient {\n private apiUrl: string\n private headers: Record<string, string>\n private credentials: 'include' | 'omit' | 'same-origin'\n\n constructor(token?: string, options: CubeApiOptions = {}) {\n this.apiUrl = options.apiUrl || '/cubejs-api/v1'\n this.headers = {\n 'Content-Type': 'application/json',\n ...options.headers\n }\n this.credentials = options.credentials ?? 'include'\n\n if (token) {\n this.headers['Authorization'] = token\n }\n }\n\n async load(query: CubeQuery, options?: { bustCache?: boolean }): Promise<CubeResultSet> {\n // Use GET with query parameter for standard Cube.js compatibility\n const queryString = JSON.stringify(query)\n const queryParam = encodeURIComponent(queryString)\n const url = `${this.apiUrl}/load?query=${queryParam}`\n\n // Build headers, optionally adding cache bust header\n const requestHeaders: Record<string, string> = {\n // Remove Content-Type for GET request\n ...Object.fromEntries(\n Object.entries(this.headers).filter(([key]) => key !== 'Content-Type')\n )\n }\n if (options?.bustCache) {\n requestHeaders['X-Cache-Control'] = 'no-cache'\n }\n\n const response = await fetch(url, {\n method: 'GET',\n headers: requestHeaders,\n credentials: this.credentials\n })\n\n if (!response.ok) {\n let errorMessage = `Cube query failed: ${response.status}`\n try {\n const errorText = await response.text()\n // Try to parse as JSON first to get structured error\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If not JSON, use the raw text\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n const result = await response.json()\n return new ResultSet(result)\n }\n\n async meta(): Promise<any> {\n const url = `${this.apiUrl}/meta`\n \n const response = await fetch(url, {\n method: 'GET',\n headers: this.headers,\n credentials: this.credentials\n })\n\n if (!response.ok) {\n throw new Error(`Failed to fetch meta: ${response.status}`)\n }\n\n return response.json()\n }\n\n async sql(query: CubeQuery): Promise<any> {\n // Use GET with query parameter for standard Cube.js compatibility\n const queryParam = encodeURIComponent(JSON.stringify(query))\n const url = `${this.apiUrl}/sql?query=${queryParam}`\n \n const response = await fetch(url, {\n method: 'GET',\n headers: {\n // Remove Content-Type for GET request\n ...Object.fromEntries(\n Object.entries(this.headers).filter(([key]) => key !== 'Content-Type')\n )\n },\n credentials: this.credentials\n })\n\n if (!response.ok) {\n throw new Error(`SQL generation failed: ${response.status}`)\n }\n\n return response.json()\n }\n\n async dryRun(query: CubeQuery): Promise<any> {\n const url = `${this.apiUrl}/dry-run`\n\n const response = await fetch(url, {\n method: 'POST',\n headers: this.headers,\n credentials: this.credentials,\n body: JSON.stringify({ query })\n })\n\n if (!response.ok) {\n let errorMessage = `Dry run failed: ${response.status}`\n try {\n const errorText = await response.text()\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n return response.json()\n }\n\n /**\n * Execute EXPLAIN on a query to get the execution plan\n * Returns normalized plan across PostgreSQL, MySQL, and SQLite\n * Accepts standard queries, funnel queries ({ funnel: {...} }), or flow queries ({ flow: {...} })\n */\n async explain(query: CubeQuery | unknown, options?: ExplainOptions): Promise<ExplainResult> {\n const url = `${this.apiUrl}/explain`\n\n const response = await fetch(url, {\n method: 'POST',\n headers: this.headers,\n credentials: this.credentials,\n body: JSON.stringify({ query, options })\n })\n\n if (!response.ok) {\n let errorMessage = `Explain failed: ${response.status}`\n try {\n const errorText = await response.text()\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n return response.json()\n }\n\n /**\n * Execute multiple queries in a single batch request\n * Used by BatchCoordinator to optimize network requests\n * Pass { bustCache: true } to bypass server-side cache\n */\n async batchLoad(queries: CubeQuery[], options?: { bustCache?: boolean }): Promise<CubeResultSet[]> {\n const url = `${this.apiUrl}/batch`\n\n // Build headers with optional cache bypass\n const requestHeaders: Record<string, string> = { ...this.headers }\n if (options?.bustCache) {\n requestHeaders['X-Cache-Control'] = 'no-cache'\n }\n\n const response = await fetch(url, {\n method: 'POST',\n headers: requestHeaders,\n credentials: this.credentials,\n body: JSON.stringify({ queries })\n })\n\n if (!response.ok) {\n let errorMessage = `Batch query failed: ${response.status}`\n try {\n const errorText = await response.text()\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n const batchResponse = await response.json()\n\n // batchResponse.results is an array of individual query results\n // Each result may have succeeded or failed\n return batchResponse.results.map((result: any) => {\n // If this individual query failed, create a ResultSet with error info\n if (!result.success && result.error) {\n // Create a result set that will throw when accessed\n return {\n ...new ResultSet({ data: [], annotation: {} }),\n error: result.error\n }\n }\n\n // Create ResultSet from successful result\n return new ResultSet(result)\n })\n }\n}\n\n/**\n * Simple ResultSet implementation\n */\nclass ResultSet implements CubeResultSet {\n public loadResponse: any\n\n constructor(loadResponse: any) {\n this.loadResponse = loadResponse\n }\n\n rawData(): any[] {\n // Handle new nested structure: loadResponse.results[0].data\n // Keep backward compatibility with old structure: loadResponse.data\n if (this.loadResponse.results && this.loadResponse.results[0]) {\n return this.loadResponse.results[0].data || []\n }\n return this.loadResponse.data || []\n }\n\n tablePivot(): any[] {\n // For pie charts and tables, return the raw data\n return this.rawData()\n }\n\n series(): any[] {\n // Simple series implementation\n return this.rawData()\n }\n\n annotation(): any {\n // Handle new nested structure: loadResponse.results[0].annotation\n // Keep backward compatibility with old structure: loadResponse.annotation\n if (this.loadResponse.results && this.loadResponse.results[0]) {\n return this.loadResponse.results[0].annotation || {}\n }\n return this.loadResponse.annotation || {}\n }\n\n /**\n * Get cache metadata if result was served from cache\n * Returns undefined if not a cache hit\n */\n cacheInfo(): { hit: true; cachedAt: string; ttlMs: number; ttlRemainingMs: number } | undefined {\n // Handle nested structure: loadResponse.results[0].cache\n if (this.loadResponse.results && this.loadResponse.results[0]) {\n return this.loadResponse.results[0].cache\n }\n return this.loadResponse.cache\n }\n}\n\n/**\n * Factory function to create a cube client\n */\nexport function createCubeClient(token?: string, options: CubeApiOptions = {}): CubeClient {\n return new CubeClient(token, options)\n}\n\n// Legacy compatibility export\nexport function cube(token?: string, options: CubeApiOptions = {}): CubeClient {\n return createCubeClient(token, options)\n}","import type { CubeQuery, CubeResultSet } from '../types'\n\n/**\n * Represents a queued query with its resolver/rejector\n */\ninterface QueuedQuery {\n query: CubeQuery\n resolve: (result: CubeResultSet) => void\n reject: (error: Error) => void\n}\n\n/**\n * BatchCoordinator collects queries triggered in the same render cycle\n * and sends them as a single batch request to minimize network overhead.\n *\n * Uses a configurable delay (default 100ms) to batch queries from lazy-loaded\n * portlets that become visible during the same scroll action.\n */\nexport class BatchCoordinator {\n private queue: QueuedQuery[] = []\n private flushScheduled = false\n private batchExecutor: (queries: CubeQuery[]) => Promise<CubeResultSet[]>\n private delayMs: number\n\n constructor(batchExecutor: (queries: CubeQuery[]) => Promise<CubeResultSet[]>, delayMs: number = 50) {\n this.batchExecutor = batchExecutor\n this.delayMs = delayMs\n }\n\n /**\n * Register a query to be batched. Returns a promise that resolves\n * when the batch is executed and this specific query's result is available.\n */\n public register(query: CubeQuery): Promise<CubeResultSet> {\n return new Promise<CubeResultSet>((resolve, reject) => {\n // Add query to queue\n this.queue.push({ query, resolve, reject })\n\n // Schedule flush if not already scheduled\n if (!this.flushScheduled) {\n this.scheduleFlush()\n }\n })\n }\n\n /**\n * Schedule a flush after a short delay to collect multiple queries.\n * The delay allows queries from lazy-loaded portlets that become visible\n * during the same scroll action to be batched together.\n */\n private scheduleFlush(): void {\n this.flushScheduled = true\n\n setTimeout(() => {\n this.flush()\n }, this.delayMs)\n }\n\n /**\n * Execute all queued queries as a batch and resolve individual promises\n */\n private async flush(): Promise<void> {\n // Reset state\n this.flushScheduled = false\n\n // Take current queue and clear it\n const currentQueue = this.queue.slice()\n this.queue = []\n\n if (currentQueue.length === 0) {\n return\n }\n\n try {\n // Extract queries\n const queries = currentQueue.map(item => item.query)\n\n // Execute batch\n const results = await this.batchExecutor(queries)\n\n // Resolve individual promises with their corresponding results\n currentQueue.forEach((item, index) => {\n const result = results[index]\n\n // Check if this specific query had an error\n if (result && 'error' in result && result.error) {\n item.reject(new Error(result.error as string))\n } else {\n item.resolve(result)\n }\n })\n } catch (error) {\n // If entire batch fails, reject all queries\n currentQueue.forEach(item => {\n item.reject(error instanceof Error ? error : new Error(String(error)))\n })\n }\n }\n\n /**\n * Get current queue size (useful for debugging)\n */\n public getQueueSize(): number {\n return this.queue.length\n }\n\n /**\n * Clear the queue (useful for testing/cleanup)\n */\n public clear(): void {\n this.queue = []\n this.flushScheduled = false\n }\n}\n","/**\n * CubeApiProvider - Stable API Context Layer\n *\n * Provides the CubeClient API instance that only changes on authentication updates.\n * Isolated from metadata and feature contexts to prevent unnecessary re-renders.\n */\n\nimport { createContext, useContext, useState, useMemo, useCallback, useEffect, type ReactNode } from 'react'\nimport { createCubeClient, type CubeClient } from '../client/CubeClient'\nimport type { CubeQueryOptions, CubeApiOptions } from '../types'\nimport { BatchCoordinator } from '../client/BatchCoordinator'\n\ninterface CubeApiContextValue {\n cubeApi: CubeClient\n options?: CubeQueryOptions\n updateApiConfig: (apiOptions: CubeApiOptions, token?: string) => void\n batchCoordinator: BatchCoordinator | null\n enableBatching: boolean\n}\n\nconst CubeApiContext = createContext<CubeApiContextValue | null>(null)\n\ninterface CubeApiProviderProps {\n apiOptions: CubeApiOptions\n token?: string\n options?: CubeQueryOptions\n enableBatching?: boolean\n batchDelayMs?: number\n children: ReactNode\n}\n\nexport function CubeApiProvider({\n apiOptions: initialApiOptions,\n token: initialToken,\n options = {},\n enableBatching = true,\n batchDelayMs = 50,\n children\n}: CubeApiProviderProps) {\n const baseConfig = useMemo(\n () => ({ apiOptions: initialApiOptions, token: initialToken }),\n [initialApiOptions, initialToken]\n )\n const [overrideConfig, setOverrideConfig] = useState<{\n apiOptions: CubeApiOptions\n token?: string\n } | null>(null)\n\n useEffect(() => {\n setOverrideConfig(null)\n }, [baseConfig])\n\n const config = overrideConfig ?? baseConfig\n\n // Create CubeClient - only recreates when config changes\n const cubeApi = useMemo(() =>\n createCubeClient(config.token, config.apiOptions),\n [config.apiOptions, config.token]\n )\n\n // Create BatchCoordinator - only recreates when cubeApi or batching config changes\n const batchCoordinator = useMemo(() => {\n if (!enableBatching) return null\n return new BatchCoordinator((queries) => cubeApi.batchLoad(queries), batchDelayMs)\n }, [enableBatching, cubeApi, batchDelayMs])\n\n // Stable callback for updating config\n const updateApiConfig = useCallback((newApiOptions: CubeApiOptions, newToken?: string) => {\n setOverrideConfig({ apiOptions: newApiOptions, token: newToken })\n }, [])\n\n // Memoize context value - only changes when cubeApi/options change\n const value = useMemo(() => ({\n cubeApi,\n options,\n updateApiConfig,\n batchCoordinator,\n enableBatching\n }), [cubeApi, options, updateApiConfig, batchCoordinator, enableBatching])\n\n return (\n <CubeApiContext.Provider value={value}>\n {children}\n </CubeApiContext.Provider>\n )\n}\n\nexport function useCubeApi() {\n const context = useContext(CubeApiContext)\n if (!context) {\n throw new Error('useCubeApi must be used within CubeApiProvider')\n }\n return context\n}\n","/**\n * useCubeMetaQuery - TanStack Query hook for cube metadata\n *\n * Replaces manual caching in useCubeMeta with TanStack Query's built-in\n * cache management. Provides:\n * - Automatic caching with configurable stale time\n * - Built-in loading and error states\n * - Automatic refetch on mount (if stale)\n * - Manual refetch capability\n *\n * This hook wraps the existing CubeClient.meta() method.\n */\n\nimport { useQuery, useQueryClient } from '@tanstack/react-query'\nimport { useCubeApi } from '../../providers/CubeApiProvider'\nimport type { CubeMeta, FieldLabelMap } from '../../types'\n\n// Query key for cube metadata\nexport const CUBE_META_QUERY_KEY = ['cube', 'meta'] as const\n\n/**\n * Build a field label map from cube metadata\n * Maps field names to their display titles\n */\nfunction buildLabelMap(meta: CubeMeta): FieldLabelMap {\n const labelMap: FieldLabelMap = {}\n\n meta.cubes.forEach((cube) => {\n // Add measures\n cube.measures.forEach((measure) => {\n labelMap[measure.name] = measure.title || measure.shortTitle || measure.name\n })\n\n // Add dimensions\n cube.dimensions.forEach((dimension) => {\n labelMap[dimension.name] = dimension.title || dimension.shortTitle || dimension.name\n })\n\n // Add segments\n cube.segments.forEach((segment) => {\n labelMap[segment.name] = segment.title || segment.shortTitle || segment.name\n })\n })\n\n return labelMap\n}\n\nexport interface UseCubeMetaQueryOptions {\n /**\n * Whether to skip the query (useful for conditional fetching)\n * @default false\n */\n enabled?: boolean\n /**\n * Stale time in milliseconds (how long before data is considered stale)\n * @default 5 * 60 * 1000 (5 minutes)\n */\n staleTime?: number\n}\n\nexport interface UseCubeMetaQueryResult {\n /** Cube metadata */\n meta: CubeMeta | null\n /** Field label map for quick lookups */\n labelMap: FieldLabelMap\n /** Whether the query is loading */\n isLoading: boolean\n /** Whether the query is fetching (includes background refetch) */\n isFetching: boolean\n /** Error if the query failed */\n error: Error | null\n /** Manually refetch the metadata */\n refetch: () => void\n /** Get a field's display label */\n getFieldLabel: (fieldName: string) => string\n}\n\n/**\n * TanStack Query hook for fetching cube metadata\n *\n * Usage:\n * ```tsx\n * const { meta, labelMap, isLoading, error, refetch } = useCubeMetaQuery()\n * ```\n */\nexport function useCubeMetaQuery(\n options: UseCubeMetaQueryOptions = {}\n): UseCubeMetaQueryResult {\n const { enabled = true, staleTime = 5 * 60 * 1000 } = options\n const { cubeApi } = useCubeApi()\n const queryClient = useQueryClient()\n\n const query = useQuery({\n queryKey: CUBE_META_QUERY_KEY,\n queryFn: async () => {\n const metaData = await cubeApi.meta()\n const labelMap = buildLabelMap(metaData)\n return { meta: metaData, labelMap }\n },\n enabled,\n staleTime,\n // Keep data in cache for 15 minutes after it becomes unused\n gcTime: 15 * 60 * 1000,\n })\n\n // Extract data from query result\n const meta = query.data?.meta ?? null\n const labelMap = query.data?.labelMap ?? {}\n\n // Stable refetch function\n const refetch = () => {\n queryClient.invalidateQueries({ queryKey: CUBE_META_QUERY_KEY })\n }\n\n // Stable getFieldLabel function\n const getFieldLabel = (fieldName: string): string => {\n return labelMap[fieldName] || fieldName\n }\n\n return {\n meta,\n labelMap,\n isLoading: query.isLoading,\n isFetching: query.isFetching,\n error: query.error,\n refetch,\n getFieldLabel,\n }\n}\n\n/**\n * Prefetch cube metadata - useful for eager loading\n *\n * Usage:\n * ```tsx\n * const queryClient = useQueryClient()\n * await prefetchCubeMeta(queryClient, cubeApi)\n * ```\n */\nexport async function prefetchCubeMeta(\n queryClient: ReturnType<typeof useQueryClient>,\n cubeApi: ReturnType<typeof useCubeApi>['cubeApi']\n): Promise<void> {\n await queryClient.prefetchQuery({\n queryKey: CUBE_META_QUERY_KEY,\n queryFn: async () => {\n const metaData = await cubeApi.meta()\n const labelMap = buildLabelMap(metaData)\n return { meta: metaData, labelMap }\n },\n })\n}\n","/**\n * Dashboard Thumbnail Capture Utility\n *\n * Provides optional screenshot functionality for dashboard thumbnails.\n * Requires modern-screenshot as an optional peer dependency.\n *\n * Usage:\n * 1. Install the optional dependency: npm install modern-screenshot\n * 2. Enable in CubeProvider: features={{ thumbnail: { enabled: true } }}\n * 3. Thumbnails are automatically captured on dashboard save\n */\n\nimport type { RefObject } from 'react'\nimport type { ThumbnailFeatureConfig } from '../types'\n\n// Type definition for modern-screenshot (optional dependency)\ntype ModernScreenshotModule = {\n domToPng: (element: HTMLElement, options?: Record<string, unknown>) => Promise<string>\n domToJpeg: (element: HTMLElement, options?: Record<string, unknown>) => Promise<string>\n}\n\n// Cache the import result to avoid repeated dynamic imports\nlet screenshotModule: ModernScreenshotModule | null = null\nlet moduleChecked = false\n\n/**\n * Resize an image to the specified dimensions while maintaining quality.\n * This takes a high-resolution source and scales it down to target dimensions,\n * which produces better results than capturing at low resolution directly.\n *\n * Uses Canvas imageSmoothingQuality for best downscaling results.\n */\nfunction resizeImage(\n dataUri: string,\n targetWidth: number,\n targetHeight: number,\n format: 'png' | 'jpeg',\n quality: number\n): Promise<string> {\n return new Promise((resolve, reject) => {\n const img = new Image()\n img.onload = () => {\n const canvas = document.createElement('canvas')\n canvas.width = targetWidth\n canvas.height = targetHeight\n\n const ctx = canvas.getContext('2d')\n if (!ctx) {\n reject(new Error('Failed to get canvas context'))\n return\n }\n\n // Enable high-quality image smoothing for downscaling\n ctx.imageSmoothingEnabled = true\n ctx.imageSmoothingQuality = 'high'\n\n // Calculate aspect-ratio-preserving dimensions\n const sourceAspect = img.width / img.height\n const targetAspect = targetWidth / targetHeight\n\n let sourceX = 0\n let sourceY = 0\n let sourceWidth = img.width\n let sourceHeight = img.height\n\n // Crop source to match target aspect ratio (center crop)\n if (sourceAspect > targetAspect) {\n // Source is wider - crop horizontally\n sourceWidth = img.height * targetAspect\n sourceX = (img.width - sourceWidth) / 2\n } else if (sourceAspect < targetAspect) {\n // Source is taller - crop vertically (take from top)\n sourceHeight = img.width / targetAspect\n // Don't center - take from top for dashboard previews\n sourceY = 0\n }\n\n // Draw the scaled and cropped image\n ctx.drawImage(\n img,\n sourceX, sourceY, sourceWidth, sourceHeight, // Source region\n 0, 0, targetWidth, targetHeight // Destination (full canvas)\n )\n\n const mimeType = format === 'jpeg' ? 'image/jpeg' : 'image/png'\n resolve(canvas.toDataURL(mimeType, quality))\n }\n img.onerror = () => reject(new Error('Failed to load image for resizing'))\n img.src = dataUri\n })\n}\n\n/**\n * Check if modern-screenshot is available (installed as peer dependency)\n */\nasync function getScreenshotModule(): Promise<ModernScreenshotModule | null> {\n if (moduleChecked) {\n return screenshotModule\n }\n\n try {\n // Dynamic import - modern-screenshot is an optional peer dependency\n // @ts-ignore - modern-screenshot may not be installed\n screenshotModule = await import('modern-screenshot') as ModernScreenshotModule\n moduleChecked = true\n return screenshotModule\n } catch {\n moduleChecked = true\n screenshotModule = null\n return null\n }\n}\n\n// Extend Window interface for our warning flag\ndeclare global {\n interface Window {\n __drizzle_cube_thumbnail_warning__?: boolean\n }\n}\n\n/**\n * Log a development-mode warning when thumbnail feature is enabled but modern-screenshot is missing\n */\nexport function warnIfScreenshotLibMissing(thumbnailConfig: ThumbnailFeatureConfig | undefined): void {\n if (typeof window === 'undefined') return\n if (!thumbnailConfig?.enabled) return\n\n // Only warn once per session\n if (window.__drizzle_cube_thumbnail_warning__) return\n\n getScreenshotModule().then((mod) => {\n if (!mod && process.env.NODE_ENV === 'development') {\n console.warn(\n '[drizzle-cube] Thumbnail feature enabled but modern-screenshot not installed. ' +\n 'Run: npm install modern-screenshot'\n )\n window.__drizzle_cube_thumbnail_warning__ = true\n }\n })\n}\n\n/**\n * Capture a thumbnail of the dashboard element\n *\n * @param elementRef - React ref to the dashboard container element\n * @param config - Thumbnail configuration (dimensions, format, quality)\n * @returns Base64 data URI of the thumbnail, or null if capture failed\n */\nexport async function captureThumbnail(\n elementRef: RefObject<HTMLElement | null>,\n config: ThumbnailFeatureConfig\n): Promise<string | null> {\n // Check if we're in a browser environment\n if (typeof window === 'undefined') {\n return null\n }\n\n // Check if the element ref is valid\n if (!elementRef.current) {\n return null\n }\n\n // Try to load modern-screenshot\n const screenshot = await getScreenshotModule()\n if (!screenshot) {\n return null\n }\n\n try {\n // Higher default dimensions for crisp thumbnails on retina displays\n // 1600x1200 provides good quality while keeping file size reasonable\n const targetWidth = config.width ?? 1600\n const targetHeight = config.height ?? 1200\n const format = config.format ?? 'png'\n const quality = config.quality ?? 0.95 // Higher default quality\n\n const element = elementRef.current\n\n // Always capture at 2x scale for high quality (like copy-to-clipboard)\n // This produces a sharp image that we then resize down\n const captureScale = 2\n\n // Get theme-aware background color (walks DOM tree to find effective bg)\n const backgroundColor = getEffectiveBackgroundColor(element)\n\n // Capture at high resolution with proper background\n const fullDataUri = await screenshot.domToPng(element, {\n scale: captureScale,\n backgroundColor,\n })\n\n // Resize to target dimensions (high-quality downscaling)\n const resizedDataUri = await resizeImage(\n fullDataUri,\n targetWidth,\n targetHeight,\n format,\n quality\n )\n\n return resizedDataUri\n } catch (error) {\n console.error('[drizzle-cube] Failed to capture thumbnail:', error)\n return null\n }\n}\n\n/**\n * Check if thumbnail capture is available and enabled\n */\nexport async function isThumbnailCaptureAvailable(\n config: ThumbnailFeatureConfig | undefined\n): Promise<boolean> {\n if (!config?.enabled) {\n return false\n }\n\n if (typeof window === 'undefined') {\n return false\n }\n\n const screenshot = await getScreenshotModule()\n return screenshot !== null\n}\n\n/**\n * Check if portlet screenshot-to-clipboard is available.\n * Requires both modern-screenshot AND Clipboard API with image support.\n */\nexport async function isPortletCopyAvailable(): Promise<boolean> {\n // Check if we're in browser environment\n if (typeof window === 'undefined') {\n return false\n }\n\n // Check modern-screenshot is installed\n const screenshot = await getScreenshotModule()\n if (!screenshot) {\n return false\n }\n\n // Check Clipboard API supports writing images (ClipboardItem with image/png blob)\n return (\n typeof ClipboardItem !== 'undefined' &&\n typeof navigator?.clipboard?.write === 'function'\n )\n}\n\n/**\n * Check if a background color is transparent or effectively invisible\n */\nfunction isTransparentBackground(color: string): boolean {\n if (!color || color === 'transparent' || color === 'rgba(0, 0, 0, 0)') {\n return true\n }\n // Check for rgba with 0 alpha\n const rgbaMatch = color.match(/rgba\\(\\s*\\d+\\s*,\\s*\\d+\\s*,\\s*\\d+\\s*,\\s*([\\d.]+)\\s*\\)/)\n if (rgbaMatch && parseFloat(rgbaMatch[1]) === 0) {\n return true\n }\n return false\n}\n\n/**\n * Find the effective background color by walking up the DOM tree\n */\nfunction getEffectiveBackgroundColor(element: HTMLElement): string {\n let current: HTMLElement | null = element\n\n while (current) {\n const bg = getComputedStyle(current).backgroundColor\n if (!isTransparentBackground(bg)) {\n return bg\n }\n current = current.parentElement\n }\n\n // Fallback to theme variable or white\n const themeColor = getComputedStyle(document.documentElement)\n .getPropertyValue('--dc-surface')\n .trim()\n\n return themeColor || '#ffffff'\n}\n\n/**\n * Capture a portlet element and copy to clipboard as PNG.\n * Returns true on success, false on failure.\n *\n * @param element - The HTML element to capture\n * @returns Promise<boolean> - true if successfully copied to clipboard\n */\nexport async function copyPortletToClipboard(element: HTMLElement): Promise<boolean> {\n try {\n const screenshot = await getScreenshotModule()\n if (!screenshot) {\n console.warn('[drizzle-cube] Cannot copy portlet: modern-screenshot not available')\n return false\n }\n\n // Get theme-aware background color by walking up the DOM tree\n const backgroundColor = getEffectiveBackgroundColor(element)\n\n // Capture as PNG data URL at 2x scale for retina quality\n const dataUrl = await screenshot.domToPng(element, {\n scale: 2,\n backgroundColor,\n })\n\n // Convert data URL to blob\n const response = await fetch(dataUrl)\n const blob = await response.blob()\n\n // Write to clipboard\n await navigator.clipboard.write([new ClipboardItem({ 'image/png': blob })])\n\n return true\n } catch (error) {\n console.warn('[drizzle-cube] Failed to copy portlet to clipboard:', error)\n return false\n }\n}\n","/**\n * CubeFeaturesProvider - Feature Flags Context Layer\n *\n * Provides feature configuration isolated from API and metadata layers.\n * This prevents components from re-rendering when feature flags change\n * if they don't use features.\n */\n\nimport { createContext, useContext, useState, useMemo, useCallback, useEffect, type ReactNode } from 'react'\nimport type { FeaturesConfig, DashboardLayoutMode } from '../types'\nimport { warnIfScreenshotLibMissing } from '../utils/thumbnail'\n\ninterface CubeFeaturesContextValue {\n features: FeaturesConfig\n dashboardModes: DashboardLayoutMode[]\n updateFeatures: (newFeatures: Partial<FeaturesConfig>) => void\n}\n\nconst CubeFeaturesContext = createContext<CubeFeaturesContextValue | null>(null)\n\ninterface CubeFeaturesProviderProps {\n features?: FeaturesConfig\n dashboardModes?: DashboardLayoutMode[]\n children: ReactNode\n}\n\nconst DEFAULT_FEATURES: FeaturesConfig = {\n enableAI: true,\n aiEndpoint: '/api/ai/generate',\n showSchemaDiagram: false,\n useAnalysisBuilder: false,\n editToolbar: 'both',\n floatingToolbarPosition: 'right'\n}\n\nexport function CubeFeaturesProvider({\n features: initialFeatures,\n dashboardModes = ['rows', 'grid'],\n children\n}: CubeFeaturesProviderProps) {\n // Merge passed features with defaults so new features get default values\n const [features, setFeatures] = useState<FeaturesConfig>(() => ({\n ...DEFAULT_FEATURES,\n ...initialFeatures\n }))\n\n // Warn in development if thumbnail feature is enabled but modern-screenshot is not installed\n useEffect(() => {\n warnIfScreenshotLibMissing(features.thumbnail)\n }, [features.thumbnail])\n\n const updateFeatures = useCallback((newFeatures: Partial<FeaturesConfig>) => {\n setFeatures(prev => ({ ...prev, ...newFeatures }))\n }, [])\n\n const value = useMemo(() => ({\n features,\n dashboardModes,\n updateFeatures\n }), [features, dashboardModes, updateFeatures])\n\n return (\n <CubeFeaturesContext.Provider value={value}>\n {children}\n </CubeFeaturesContext.Provider>\n )\n}\n\n// Default context value when used outside provider (safe fallback for hooks)\nconst DEFAULT_CONTEXT: CubeFeaturesContextValue = {\n features: DEFAULT_FEATURES,\n dashboardModes: ['rows', 'grid'],\n updateFeatures: () => {\n // No-op when used outside provider\n }\n}\n\n/**\n * Hook to access cube features context.\n * Returns default values if used outside CubeFeaturesProvider (graceful fallback).\n */\nexport function useCubeFeatures() {\n const context = useContext(CubeFeaturesContext)\n // Return default context if not within provider (allows hooks like useCubeLoadQuery to work in isolation)\n return context ?? DEFAULT_CONTEXT\n}\n","/**\n * ScrollContainerContext\n *\n * Provides the scroll container element for lazy loading with IntersectionObserver.\n * This allows portlets to detect visibility relative to a custom scroll container\n * (not just the viewport) when the dashboard is embedded in a scrolling div.\n */\n\nimport { createContext, useContext } from 'react'\n\nconst ScrollContainerContext = createContext<HTMLElement | null>(null)\n\n/**\n * Provider component to wrap dashboard content with a scroll container reference.\n * Used by DashboardGrid and MobileStackedLayout to pass the detected scroll container\n * to child portlets.\n */\nexport const ScrollContainerProvider = ScrollContainerContext.Provider\n\n/**\n * Hook to access the scroll container element for lazy loading.\n * Returns null if using viewport (window) scroll, or the container element\n * if the dashboard is inside a scrolling container.\n */\nexport const useScrollContainer = () => useContext(ScrollContainerContext)\n","/**\n * CubeMetaContext - Metadata context definitions.\n *\n * Split from CubeMetaProvider so consumers can access metadata context\n * without pulling in TanStack Query dependencies.\n */\n\nimport { createContext, useContext } from 'react'\nimport type { CubeMeta, FieldLabelMap } from '../types'\n\nexport interface CubeMetaContextValue {\n meta: CubeMeta | null\n labelMap: FieldLabelMap\n metaLoading: boolean\n metaError: string | null\n getFieldLabel: (fieldName: string) => string\n refetchMeta: () => void\n}\n\nexport const CubeMetaContext = createContext<CubeMetaContextValue | null>(null)\n\nexport function useCubeMeta() {\n const context = useContext(CubeMetaContext)\n if (!context) {\n throw new Error('useCubeMeta must be used within CubeMetaProvider')\n }\n return context\n}\n","/**\n * CubeMetaProvider - Metadata Context Layer\n *\n * Provides cube metadata (meta, labelMap) isolated from API layer.\n * This prevents components from re-rendering when metadata loads\n * if they only need API access.\n *\n * Uses TanStack Query for metadata fetching with built-in caching.\n */\n\nimport { useCallback, useMemo, type ReactNode } from 'react'\nimport { useCubeMetaQuery } from '../hooks/queries/useCubeMetaQuery'\nimport { CubeMetaContext } from './CubeMetaContext'\n\ninterface CubeMetaProviderProps {\n children: ReactNode\n}\n\nexport function CubeMetaProvider({ children }: CubeMetaProviderProps) {\n // Use TanStack Query hook for metadata fetching\n const {\n meta,\n labelMap,\n isLoading: metaLoading,\n error,\n refetch,\n getFieldLabel: getFieldLabelFromQuery\n } = useCubeMetaQuery()\n\n // Convert error to string for backward compatibility\n const metaError = error ? (error instanceof Error ? error.message : String(error)) : null\n\n // Wrap refetch to match expected signature\n const refetchMeta = useCallback(() => {\n refetch()\n }, [refetch])\n\n // Create stable getFieldLabel that reads from current labelMap\n const getFieldLabel = useCallback((fieldName: string): string => {\n return getFieldLabelFromQuery(fieldName)\n }, [getFieldLabelFromQuery])\n\n // Memoize context value to prevent unnecessary re-renders\n const value = useMemo(() => ({\n meta,\n labelMap,\n metaLoading,\n metaError,\n getFieldLabel,\n refetchMeta\n }), [meta, labelMap, metaLoading, metaError, getFieldLabel, refetchMeta])\n\n return (\n <CubeMetaContext.Provider value={value}>\n {children}\n </CubeMetaContext.Provider>\n )\n}\n","/**\n * CubeProvider - Backward Compatibility Wrapper\n *\n * This provider wraps the three specialized context providers (API, Meta, Features)\n * to maintain 100% backward compatibility with existing code using useCubeContext().\n *\n * New code should use the specialized hooks (useCubeApi, useCubeMeta, useCubeFeatures)\n * for better performance and selective re-rendering.\n *\n * Also includes TanStack Query's QueryClientProvider for data fetching in\n * AnalysisBuilder and other components that use TanStack Query hooks.\n */\n\nimport { useMemo, useState, type ReactNode } from 'react'\nimport { QueryClient, QueryClientProvider } from '@tanstack/react-query'\nimport { CubeApiProvider, useCubeApi } from './CubeApiProvider'\nimport { CubeMetaProvider } from './CubeMetaProvider'\nimport { useCubeMeta } from './CubeMetaContext'\nimport { CubeFeaturesProvider, useCubeFeatures } from './CubeFeaturesProvider'\nimport type { CubeQueryOptions, CubeApiOptions, FeaturesConfig, DashboardLayoutMode, CubeMeta, FieldLabelMap } from '../types'\nimport type { CubeClient } from '../client/CubeClient'\nimport type { BatchCoordinator } from '../client/BatchCoordinator'\n\nconst DEFAULT_API_OPTIONS: CubeApiOptions = { apiUrl: '/cubejs-api/v1' }\n\nexport const createCubeQueryClient = () => new QueryClient({\n defaultOptions: {\n queries: {\n // Stale time: 5 minutes (matches existing cache duration)\n staleTime: 5 * 60 * 1000,\n // GC time: 15 minutes\n gcTime: 15 * 60 * 1000,\n // Retry failed queries up to 3 times\n retry: 3,\n // Don't refetch on window focus by default (can be overridden per-query)\n refetchOnWindowFocus: false,\n },\n },\n})\n\n// Legacy default client (useful for tests or explicit injection)\nexport const queryClient = createCubeQueryClient()\n\n// Backward compatible interface - merges all three contexts\ninterface CubeContextValue {\n cubeApi: CubeClient\n options?: CubeQueryOptions\n meta: CubeMeta | null\n labelMap: FieldLabelMap\n metaLoading: boolean\n metaError: string | null\n getFieldLabel: (fieldName: string) => string\n refetchMeta: () => void\n updateApiConfig: (apiOptions: CubeApiOptions, token?: string) => void\n features: FeaturesConfig\n batchCoordinator: BatchCoordinator | null\n enableBatching: boolean\n dashboardModes: DashboardLayoutMode[]\n}\n\ninterface CubeProviderProps {\n cubeApi?: CubeClient\n apiOptions?: CubeApiOptions\n token?: string\n options?: CubeQueryOptions\n features?: FeaturesConfig\n dashboardModes?: DashboardLayoutMode[]\n enableBatching?: boolean\n batchDelayMs?: number // Delay in ms to collect queries before batching (default: 100)\n queryClient?: QueryClient\n children: ReactNode\n}\n\n/**\n * CubeProvider - Three-layer context wrapper\n *\n * Wraps children in three isolated context providers for optimal performance:\n * 1. CubeApiProvider - Stable API layer (changes only on auth)\n * 2. CubeMetaProvider - Metadata layer (changes on metadata load)\n * 3. CubeFeaturesProvider - Feature flags layer (changes on feature updates)\n */\nexport function CubeProvider({\n cubeApi: _initialCubeApi, // Intentionally unused - for backward compatibility\n apiOptions,\n token,\n options,\n features,\n dashboardModes,\n enableBatching,\n batchDelayMs,\n queryClient: providedQueryClient,\n children\n}: CubeProviderProps) {\n const [internalQueryClient] = useState(() => createCubeQueryClient())\n const queryClient = providedQueryClient ?? internalQueryClient\n\n return (\n <QueryClientProvider client={queryClient}>\n <CubeApiProvider\n apiOptions={apiOptions || DEFAULT_API_OPTIONS}\n token={token}\n options={options}\n enableBatching={enableBatching}\n batchDelayMs={batchDelayMs}\n >\n <CubeMetaProvider>\n <CubeFeaturesProvider features={features} dashboardModes={dashboardModes}>\n {children}\n </CubeFeaturesProvider>\n </CubeMetaProvider>\n </CubeApiProvider>\n </QueryClientProvider>\n )\n}\n\n/**\n * useCubeContext - Backward compatible hook\n *\n * Merges all three contexts into a single object for backward compatibility.\n * Components using this hook will re-render when ANY context changes.\n *\n * For better performance, use specialized hooks:\n * - useCubeApi() - Only re-renders on API changes\n * - useCubeMeta() - Only re-renders on metadata changes\n * - useCubeFeatures() - Only re-renders on feature changes\n */\nexport function useCubeContext(): CubeContextValue {\n const api = useCubeApi()\n const meta = useCubeMeta()\n const featuresCtx = useCubeFeatures()\n\n return useMemo(() => ({\n ...api,\n ...meta,\n features: featuresCtx.features,\n dashboardModes: featuresCtx.dashboardModes\n }), [api, meta, featuresCtx])\n}\n\n// Re-export specialized hooks for better tree-shaking and performance\nexport { useCubeApi, useCubeMeta, useCubeFeatures }\n\n// Export factory for testing and advanced use cases\nexport { createCubeQueryClient as createQueryClient }\n"],"names":["CubeClient","token","options","query","queryString","queryParam","url","requestHeaders","key","response","errorMessage","errorText","errorData","result","ResultSet","queries","loadResponse","createCubeClient","BatchCoordinator","batchExecutor","delayMs","resolve","reject","currentQueue","item","results","index","error","CubeApiContext","createContext","CubeApiProvider","initialApiOptions","initialToken","enableBatching","batchDelayMs","children","baseConfig","useMemo","overrideConfig","setOverrideConfig","useState","useEffect","config","cubeApi","batchCoordinator","updateApiConfig","useCallback","newApiOptions","newToken","value","jsx","useCubeApi","context","useContext","CUBE_META_QUERY_KEY","buildLabelMap","meta","labelMap","cube","measure","dimension","segment","useCubeMetaQuery","enabled","staleTime","queryClient","useQueryClient","useQuery","metaData","refetch","getFieldLabel","fieldName","screenshotModule","moduleChecked","resizeImage","dataUri","targetWidth","targetHeight","format","quality","img","canvas","ctx","sourceAspect","targetAspect","sourceX","sourceY","sourceWidth","sourceHeight","mimeType","getScreenshotModule","warnIfScreenshotLibMissing","thumbnailConfig","mod","captureThumbnail","elementRef","screenshot","element","captureScale","backgroundColor","getEffectiveBackgroundColor","fullDataUri","isPortletCopyAvailable","isTransparentBackground","color","rgbaMatch","current","bg","copyPortletToClipboard","dataUrl","blob","CubeFeaturesContext","DEFAULT_FEATURES","CubeFeaturesProvider","initialFeatures","dashboardModes","features","setFeatures","updateFeatures","newFeatures","prev","DEFAULT_CONTEXT","useCubeFeatures","ScrollContainerContext","ScrollContainerProvider","useScrollContainer","CubeMetaContext","useCubeMeta","CubeMetaProvider","metaLoading","getFieldLabelFromQuery","metaError","refetchMeta","DEFAULT_API_OPTIONS","createCubeQueryClient","QueryClient","CubeProvider","_initialCubeApi","apiOptions","providedQueryClient","internalQueryClient","QueryClientProvider","useCubeContext","api","featuresCtx"],"mappings":"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"}
1
+ {"version":3,"file":"providers-BW8D7Wso.js","sources":["../../../src/client/client/CubeClient.ts","../../../src/client/client/BatchCoordinator.ts","../../../src/client/providers/CubeApiProvider.tsx","../../../src/client/hooks/queries/useCubeMetaQuery.ts","../../../src/client/utils/thumbnail.ts","../../../src/client/providers/CubeFeaturesProvider.tsx","../../../src/client/providers/ScrollContainerContext.tsx","../../../src/client/providers/CubeMetaContext.tsx","../../../src/client/providers/CubeMetaProvider.tsx","../../../src/client/providers/CubeProvider.tsx"],"sourcesContent":["/**\n * Minimal Cube client implementation\n * Replaces @cubejs-client/core with lighter implementation\n */\n\nimport type { CubeQuery, CubeApiOptions, CubeResultSet, ExplainResult, ExplainOptions } from '../types'\n\nexport class CubeClient {\n private apiUrl: string\n private headers: Record<string, string>\n private credentials: 'include' | 'omit' | 'same-origin'\n\n constructor(token?: string, options: CubeApiOptions = {}) {\n this.apiUrl = options.apiUrl || '/cubejs-api/v1'\n this.headers = {\n 'Content-Type': 'application/json',\n ...options.headers\n }\n this.credentials = options.credentials ?? 'include'\n\n if (token) {\n this.headers['Authorization'] = token\n }\n }\n\n async load(query: CubeQuery, options?: { bustCache?: boolean }): Promise<CubeResultSet> {\n // Use GET with query parameter for standard Cube.js compatibility\n const queryString = JSON.stringify(query)\n const queryParam = encodeURIComponent(queryString)\n const url = `${this.apiUrl}/load?query=${queryParam}`\n\n // Build headers, optionally adding cache bust header\n const requestHeaders: Record<string, string> = {\n // Remove Content-Type for GET request\n ...Object.fromEntries(\n Object.entries(this.headers).filter(([key]) => key !== 'Content-Type')\n )\n }\n if (options?.bustCache) {\n requestHeaders['X-Cache-Control'] = 'no-cache'\n }\n\n const response = await fetch(url, {\n method: 'GET',\n headers: requestHeaders,\n credentials: this.credentials\n })\n\n if (!response.ok) {\n let errorMessage = `Cube query failed: ${response.status}`\n try {\n const errorText = await response.text()\n // Try to parse as JSON first to get structured error\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If not JSON, use the raw text\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n const result = await response.json()\n return new ResultSet(result)\n }\n\n async meta(): Promise<any> {\n const url = `${this.apiUrl}/meta`\n \n const response = await fetch(url, {\n method: 'GET',\n headers: this.headers,\n credentials: this.credentials\n })\n\n if (!response.ok) {\n throw new Error(`Failed to fetch meta: ${response.status}`)\n }\n\n return response.json()\n }\n\n async sql(query: CubeQuery): Promise<any> {\n // Use GET with query parameter for standard Cube.js compatibility\n const queryParam = encodeURIComponent(JSON.stringify(query))\n const url = `${this.apiUrl}/sql?query=${queryParam}`\n \n const response = await fetch(url, {\n method: 'GET',\n headers: {\n // Remove Content-Type for GET request\n ...Object.fromEntries(\n Object.entries(this.headers).filter(([key]) => key !== 'Content-Type')\n )\n },\n credentials: this.credentials\n })\n\n if (!response.ok) {\n throw new Error(`SQL generation failed: ${response.status}`)\n }\n\n return response.json()\n }\n\n async dryRun(query: CubeQuery): Promise<any> {\n const url = `${this.apiUrl}/dry-run`\n\n const response = await fetch(url, {\n method: 'POST',\n headers: this.headers,\n credentials: this.credentials,\n body: JSON.stringify({ query })\n })\n\n if (!response.ok) {\n let errorMessage = `Dry run failed: ${response.status}`\n try {\n const errorText = await response.text()\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n return response.json()\n }\n\n /**\n * Execute EXPLAIN on a query to get the execution plan\n * Returns normalized plan across PostgreSQL, MySQL, and SQLite\n * Accepts standard queries, funnel queries ({ funnel: {...} }), or flow queries ({ flow: {...} })\n */\n async explain(query: CubeQuery | unknown, options?: ExplainOptions): Promise<ExplainResult> {\n const url = `${this.apiUrl}/explain`\n\n const response = await fetch(url, {\n method: 'POST',\n headers: this.headers,\n credentials: this.credentials,\n body: JSON.stringify({ query, options })\n })\n\n if (!response.ok) {\n let errorMessage = `Explain failed: ${response.status}`\n try {\n const errorText = await response.text()\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n return response.json()\n }\n\n /**\n * Execute multiple queries in a single batch request\n * Used by BatchCoordinator to optimize network requests\n * Pass { bustCache: true } to bypass server-side cache\n */\n async batchLoad(queries: CubeQuery[], options?: { bustCache?: boolean }): Promise<CubeResultSet[]> {\n const url = `${this.apiUrl}/batch`\n\n // Build headers with optional cache bypass\n const requestHeaders: Record<string, string> = { ...this.headers }\n if (options?.bustCache) {\n requestHeaders['X-Cache-Control'] = 'no-cache'\n }\n\n const response = await fetch(url, {\n method: 'POST',\n headers: requestHeaders,\n credentials: this.credentials,\n body: JSON.stringify({ queries })\n })\n\n if (!response.ok) {\n let errorMessage = `Batch query failed: ${response.status}`\n try {\n const errorText = await response.text()\n try {\n const errorData = JSON.parse(errorText)\n if (errorData.error) {\n errorMessage = errorData.error\n } else {\n errorMessage += ` ${errorText}`\n }\n } catch {\n errorMessage += ` ${errorText}`\n }\n } catch {\n // If we can't read the response, just use the status\n }\n throw new Error(errorMessage)\n }\n\n const batchResponse = await response.json()\n\n // batchResponse.results is an array of individual query results\n // Each result may have succeeded or failed\n return batchResponse.results.map((result: any) => {\n // If this individual query failed, create a ResultSet with error info\n if (!result.success && result.error) {\n // Create a result set that will throw when accessed\n return {\n ...new ResultSet({ data: [], annotation: {} }),\n error: result.error\n }\n }\n\n // Create ResultSet from successful result\n return new ResultSet(result)\n })\n }\n}\n\n/**\n * Simple ResultSet implementation\n */\nclass ResultSet implements CubeResultSet {\n public loadResponse: any\n\n constructor(loadResponse: any) {\n this.loadResponse = loadResponse\n }\n\n rawData(): any[] {\n // Handle new nested structure: loadResponse.results[0].data\n // Keep backward compatibility with old structure: loadResponse.data\n if (this.loadResponse.results && this.loadResponse.results[0]) {\n return this.loadResponse.results[0].data || []\n }\n return this.loadResponse.data || []\n }\n\n tablePivot(): any[] {\n // For pie charts and tables, return the raw data\n return this.rawData()\n }\n\n series(): any[] {\n // Simple series implementation\n return this.rawData()\n }\n\n annotation(): any {\n // Handle new nested structure: loadResponse.results[0].annotation\n // Keep backward compatibility with old structure: loadResponse.annotation\n if (this.loadResponse.results && this.loadResponse.results[0]) {\n return this.loadResponse.results[0].annotation || {}\n }\n return this.loadResponse.annotation || {}\n }\n\n /**\n * Get cache metadata if result was served from cache\n * Returns undefined if not a cache hit\n */\n cacheInfo(): { hit: true; cachedAt: string; ttlMs: number; ttlRemainingMs: number } | undefined {\n // Handle nested structure: loadResponse.results[0].cache\n if (this.loadResponse.results && this.loadResponse.results[0]) {\n return this.loadResponse.results[0].cache\n }\n return this.loadResponse.cache\n }\n}\n\n/**\n * Factory function to create a cube client\n */\nexport function createCubeClient(token?: string, options: CubeApiOptions = {}): CubeClient {\n return new CubeClient(token, options)\n}\n\n// Legacy compatibility export\nexport function cube(token?: string, options: CubeApiOptions = {}): CubeClient {\n return createCubeClient(token, options)\n}","import type { CubeQuery, CubeResultSet } from '../types'\n\n/**\n * Represents a queued query with its resolver/rejector\n */\ninterface QueuedQuery {\n query: CubeQuery\n resolve: (result: CubeResultSet) => void\n reject: (error: Error) => void\n}\n\n/**\n * BatchCoordinator collects queries triggered in the same render cycle\n * and sends them as a single batch request to minimize network overhead.\n *\n * Uses a configurable delay (default 100ms) to batch queries from lazy-loaded\n * portlets that become visible during the same scroll action.\n */\nexport class BatchCoordinator {\n private queue: QueuedQuery[] = []\n private flushScheduled = false\n private batchExecutor: (queries: CubeQuery[]) => Promise<CubeResultSet[]>\n private delayMs: number\n\n constructor(batchExecutor: (queries: CubeQuery[]) => Promise<CubeResultSet[]>, delayMs: number = 50) {\n this.batchExecutor = batchExecutor\n this.delayMs = delayMs\n }\n\n /**\n * Register a query to be batched. Returns a promise that resolves\n * when the batch is executed and this specific query's result is available.\n */\n public register(query: CubeQuery): Promise<CubeResultSet> {\n return new Promise<CubeResultSet>((resolve, reject) => {\n // Add query to queue\n this.queue.push({ query, resolve, reject })\n\n // Schedule flush if not already scheduled\n if (!this.flushScheduled) {\n this.scheduleFlush()\n }\n })\n }\n\n /**\n * Schedule a flush after a short delay to collect multiple queries.\n * The delay allows queries from lazy-loaded portlets that become visible\n * during the same scroll action to be batched together.\n */\n private scheduleFlush(): void {\n this.flushScheduled = true\n\n setTimeout(() => {\n this.flush()\n }, this.delayMs)\n }\n\n /**\n * Execute all queued queries as a batch and resolve individual promises\n */\n private async flush(): Promise<void> {\n // Reset state\n this.flushScheduled = false\n\n // Take current queue and clear it\n const currentQueue = this.queue.slice()\n this.queue = []\n\n if (currentQueue.length === 0) {\n return\n }\n\n try {\n // Extract queries\n const queries = currentQueue.map(item => item.query)\n\n // Execute batch\n const results = await this.batchExecutor(queries)\n\n // Resolve individual promises with their corresponding results\n currentQueue.forEach((item, index) => {\n const result = results[index]\n\n // Check if this specific query had an error\n if (result && 'error' in result && result.error) {\n item.reject(new Error(result.error as string))\n } else {\n item.resolve(result)\n }\n })\n } catch (error) {\n // If entire batch fails, reject all queries\n currentQueue.forEach(item => {\n item.reject(error instanceof Error ? error : new Error(String(error)))\n })\n }\n }\n\n /**\n * Get current queue size (useful for debugging)\n */\n public getQueueSize(): number {\n return this.queue.length\n }\n\n /**\n * Clear the queue (useful for testing/cleanup)\n */\n public clear(): void {\n this.queue = []\n this.flushScheduled = false\n }\n}\n","/**\n * CubeApiProvider - Stable API Context Layer\n *\n * Provides the CubeClient API instance that only changes on authentication updates.\n * Isolated from metadata and feature contexts to prevent unnecessary re-renders.\n */\n\nimport { createContext, useContext, useState, useMemo, useCallback, useEffect, type ReactNode } from 'react'\nimport { createCubeClient, type CubeClient } from '../client/CubeClient'\nimport type { CubeQueryOptions, CubeApiOptions } from '../types'\nimport { BatchCoordinator } from '../client/BatchCoordinator'\n\ninterface CubeApiContextValue {\n cubeApi: CubeClient\n options?: CubeQueryOptions\n updateApiConfig: (apiOptions: CubeApiOptions, token?: string) => void\n batchCoordinator: BatchCoordinator | null\n enableBatching: boolean\n}\n\nconst CubeApiContext = createContext<CubeApiContextValue | null>(null)\n\ninterface CubeApiProviderProps {\n apiOptions: CubeApiOptions\n token?: string\n options?: CubeQueryOptions\n enableBatching?: boolean\n batchDelayMs?: number\n children: ReactNode\n}\n\nexport function CubeApiProvider({\n apiOptions: initialApiOptions,\n token: initialToken,\n options = {},\n enableBatching = true,\n batchDelayMs = 50,\n children\n}: CubeApiProviderProps) {\n const baseConfig = useMemo(\n () => ({ apiOptions: initialApiOptions, token: initialToken }),\n [initialApiOptions, initialToken]\n )\n const [overrideConfig, setOverrideConfig] = useState<{\n apiOptions: CubeApiOptions\n token?: string\n } | null>(null)\n\n useEffect(() => {\n setOverrideConfig(null)\n }, [baseConfig])\n\n const config = overrideConfig ?? baseConfig\n\n // Create CubeClient - only recreates when config changes\n const cubeApi = useMemo(() =>\n createCubeClient(config.token, config.apiOptions),\n [config.apiOptions, config.token]\n )\n\n // Create BatchCoordinator - only recreates when cubeApi or batching config changes\n const batchCoordinator = useMemo(() => {\n if (!enableBatching) return null\n return new BatchCoordinator((queries) => cubeApi.batchLoad(queries), batchDelayMs)\n }, [enableBatching, cubeApi, batchDelayMs])\n\n // Stable callback for updating config\n const updateApiConfig = useCallback((newApiOptions: CubeApiOptions, newToken?: string) => {\n setOverrideConfig({ apiOptions: newApiOptions, token: newToken })\n }, [])\n\n // Memoize context value - only changes when cubeApi/options change\n const value = useMemo(() => ({\n cubeApi,\n options,\n updateApiConfig,\n batchCoordinator,\n enableBatching\n }), [cubeApi, options, updateApiConfig, batchCoordinator, enableBatching])\n\n return (\n <CubeApiContext.Provider value={value}>\n {children}\n </CubeApiContext.Provider>\n )\n}\n\nexport function useCubeApi() {\n const context = useContext(CubeApiContext)\n if (!context) {\n throw new Error('useCubeApi must be used within CubeApiProvider')\n }\n return context\n}\n","/**\n * useCubeMetaQuery - TanStack Query hook for cube metadata\n *\n * Replaces manual caching in useCubeMeta with TanStack Query's built-in\n * cache management. Provides:\n * - Automatic caching with configurable stale time\n * - Built-in loading and error states\n * - Automatic refetch on mount (if stale)\n * - Manual refetch capability\n *\n * This hook wraps the existing CubeClient.meta() method.\n */\n\nimport { useQuery, useQueryClient } from '@tanstack/react-query'\nimport { useCubeApi } from '../../providers/CubeApiProvider'\nimport type { CubeMeta, FieldLabelMap } from '../../types'\n\n// Query key for cube metadata\nexport const CUBE_META_QUERY_KEY = ['cube', 'meta'] as const\n\n/**\n * Build a field label map from cube metadata\n * Maps field names to their display titles\n */\nfunction buildLabelMap(meta: CubeMeta): FieldLabelMap {\n const labelMap: FieldLabelMap = {}\n\n meta.cubes.forEach((cube) => {\n // Add measures\n cube.measures.forEach((measure) => {\n labelMap[measure.name] = measure.title || measure.shortTitle || measure.name\n })\n\n // Add dimensions\n cube.dimensions.forEach((dimension) => {\n labelMap[dimension.name] = dimension.title || dimension.shortTitle || dimension.name\n })\n\n // Add segments\n cube.segments.forEach((segment) => {\n labelMap[segment.name] = segment.title || segment.shortTitle || segment.name\n })\n })\n\n return labelMap\n}\n\nexport interface UseCubeMetaQueryOptions {\n /**\n * Whether to skip the query (useful for conditional fetching)\n * @default false\n */\n enabled?: boolean\n /**\n * Stale time in milliseconds (how long before data is considered stale)\n * @default 5 * 60 * 1000 (5 minutes)\n */\n staleTime?: number\n}\n\nexport interface UseCubeMetaQueryResult {\n /** Cube metadata */\n meta: CubeMeta | null\n /** Field label map for quick lookups */\n labelMap: FieldLabelMap\n /** Whether the query is loading */\n isLoading: boolean\n /** Whether the query is fetching (includes background refetch) */\n isFetching: boolean\n /** Error if the query failed */\n error: Error | null\n /** Manually refetch the metadata */\n refetch: () => void\n /** Get a field's display label */\n getFieldLabel: (fieldName: string) => string\n}\n\n/**\n * TanStack Query hook for fetching cube metadata\n *\n * Usage:\n * ```tsx\n * const { meta, labelMap, isLoading, error, refetch } = useCubeMetaQuery()\n * ```\n */\nexport function useCubeMetaQuery(\n options: UseCubeMetaQueryOptions = {}\n): UseCubeMetaQueryResult {\n const { enabled = true, staleTime = 5 * 60 * 1000 } = options\n const { cubeApi } = useCubeApi()\n const queryClient = useQueryClient()\n\n const query = useQuery({\n queryKey: CUBE_META_QUERY_KEY,\n queryFn: async () => {\n const metaData = await cubeApi.meta()\n const labelMap = buildLabelMap(metaData)\n return { meta: metaData, labelMap }\n },\n enabled,\n staleTime,\n // Keep data in cache for 15 minutes after it becomes unused\n gcTime: 15 * 60 * 1000,\n })\n\n // Extract data from query result\n const meta = query.data?.meta ?? null\n const labelMap = query.data?.labelMap ?? {}\n\n // Stable refetch function\n const refetch = () => {\n queryClient.invalidateQueries({ queryKey: CUBE_META_QUERY_KEY })\n }\n\n // Stable getFieldLabel function\n const getFieldLabel = (fieldName: string): string => {\n return labelMap[fieldName] || fieldName\n }\n\n return {\n meta,\n labelMap,\n isLoading: query.isLoading,\n isFetching: query.isFetching,\n error: query.error,\n refetch,\n getFieldLabel,\n }\n}\n\n/**\n * Prefetch cube metadata - useful for eager loading\n *\n * Usage:\n * ```tsx\n * const queryClient = useQueryClient()\n * await prefetchCubeMeta(queryClient, cubeApi)\n * ```\n */\nexport async function prefetchCubeMeta(\n queryClient: ReturnType<typeof useQueryClient>,\n cubeApi: ReturnType<typeof useCubeApi>['cubeApi']\n): Promise<void> {\n await queryClient.prefetchQuery({\n queryKey: CUBE_META_QUERY_KEY,\n queryFn: async () => {\n const metaData = await cubeApi.meta()\n const labelMap = buildLabelMap(metaData)\n return { meta: metaData, labelMap }\n },\n })\n}\n","/**\n * Dashboard Thumbnail Capture Utility\n *\n * Provides optional screenshot functionality for dashboard thumbnails.\n * Requires modern-screenshot as an optional peer dependency.\n *\n * Usage:\n * 1. Install the optional dependency: npm install modern-screenshot\n * 2. Enable in CubeProvider: features={{ thumbnail: { enabled: true } }}\n * 3. Thumbnails are automatically captured on dashboard save\n */\n\nimport type { RefObject } from 'react'\nimport type { ThumbnailFeatureConfig } from '../types'\n\n// Type definition for modern-screenshot (optional dependency)\ntype ModernScreenshotModule = {\n domToPng: (element: HTMLElement, options?: Record<string, unknown>) => Promise<string>\n domToJpeg: (element: HTMLElement, options?: Record<string, unknown>) => Promise<string>\n}\n\n// Cache the import result to avoid repeated dynamic imports\nlet screenshotModule: ModernScreenshotModule | null = null\nlet moduleChecked = false\n\n/**\n * Resize an image to the specified dimensions while maintaining quality.\n * This takes a high-resolution source and scales it down to target dimensions,\n * which produces better results than capturing at low resolution directly.\n *\n * Uses Canvas imageSmoothingQuality for best downscaling results.\n */\nfunction resizeImage(\n dataUri: string,\n targetWidth: number,\n targetHeight: number,\n format: 'png' | 'jpeg',\n quality: number\n): Promise<string> {\n return new Promise((resolve, reject) => {\n const img = new Image()\n img.onload = () => {\n const canvas = document.createElement('canvas')\n canvas.width = targetWidth\n canvas.height = targetHeight\n\n const ctx = canvas.getContext('2d')\n if (!ctx) {\n reject(new Error('Failed to get canvas context'))\n return\n }\n\n // Enable high-quality image smoothing for downscaling\n ctx.imageSmoothingEnabled = true\n ctx.imageSmoothingQuality = 'high'\n\n // Calculate aspect-ratio-preserving dimensions\n const sourceAspect = img.width / img.height\n const targetAspect = targetWidth / targetHeight\n\n let sourceX = 0\n let sourceY = 0\n let sourceWidth = img.width\n let sourceHeight = img.height\n\n // Crop source to match target aspect ratio (center crop)\n if (sourceAspect > targetAspect) {\n // Source is wider - crop horizontally\n sourceWidth = img.height * targetAspect\n sourceX = (img.width - sourceWidth) / 2\n } else if (sourceAspect < targetAspect) {\n // Source is taller - crop vertically (take from top)\n sourceHeight = img.width / targetAspect\n // Don't center - take from top for dashboard previews\n sourceY = 0\n }\n\n // Draw the scaled and cropped image\n ctx.drawImage(\n img,\n sourceX, sourceY, sourceWidth, sourceHeight, // Source region\n 0, 0, targetWidth, targetHeight // Destination (full canvas)\n )\n\n const mimeType = format === 'jpeg' ? 'image/jpeg' : 'image/png'\n resolve(canvas.toDataURL(mimeType, quality))\n }\n img.onerror = () => reject(new Error('Failed to load image for resizing'))\n img.src = dataUri\n })\n}\n\n/**\n * Check if modern-screenshot is available (installed as peer dependency)\n */\nasync function getScreenshotModule(): Promise<ModernScreenshotModule | null> {\n if (moduleChecked) {\n return screenshotModule\n }\n\n try {\n // Dynamic import - modern-screenshot is an optional peer dependency\n // @ts-ignore - modern-screenshot may not be installed\n screenshotModule = await import('modern-screenshot') as ModernScreenshotModule\n moduleChecked = true\n return screenshotModule\n } catch {\n moduleChecked = true\n screenshotModule = null\n return null\n }\n}\n\n// Extend Window interface for our warning flag\ndeclare global {\n interface Window {\n __drizzle_cube_thumbnail_warning__?: boolean\n }\n}\n\n/**\n * Log a development-mode warning when thumbnail feature is enabled but modern-screenshot is missing\n */\nexport function warnIfScreenshotLibMissing(thumbnailConfig: ThumbnailFeatureConfig | undefined): void {\n if (typeof window === 'undefined') return\n if (!thumbnailConfig?.enabled) return\n\n // Only warn once per session\n if (window.__drizzle_cube_thumbnail_warning__) return\n\n getScreenshotModule().then((mod) => {\n if (!mod && process.env.NODE_ENV === 'development') {\n console.warn(\n '[drizzle-cube] Thumbnail feature enabled but modern-screenshot not installed. ' +\n 'Run: npm install modern-screenshot'\n )\n window.__drizzle_cube_thumbnail_warning__ = true\n }\n })\n}\n\n/**\n * Capture a thumbnail of the dashboard element\n *\n * @param elementRef - React ref to the dashboard container element\n * @param config - Thumbnail configuration (dimensions, format, quality)\n * @returns Base64 data URI of the thumbnail, or null if capture failed\n */\nexport async function captureThumbnail(\n elementRef: RefObject<HTMLElement | null>,\n config: ThumbnailFeatureConfig\n): Promise<string | null> {\n // Check if we're in a browser environment\n if (typeof window === 'undefined') {\n return null\n }\n\n // Check if the element ref is valid\n if (!elementRef.current) {\n return null\n }\n\n // Try to load modern-screenshot\n const screenshot = await getScreenshotModule()\n if (!screenshot) {\n return null\n }\n\n try {\n // Higher default dimensions for crisp thumbnails on retina displays\n // 1600x1200 provides good quality while keeping file size reasonable\n const targetWidth = config.width ?? 1600\n const targetHeight = config.height ?? 1200\n const format = config.format ?? 'png'\n const quality = config.quality ?? 0.95 // Higher default quality\n\n const element = elementRef.current\n\n // Always capture at 2x scale for high quality (like copy-to-clipboard)\n // This produces a sharp image that we then resize down\n const captureScale = 2\n\n // Get theme-aware background color (walks DOM tree to find effective bg)\n const backgroundColor = getEffectiveBackgroundColor(element)\n\n // Capture at high resolution with proper background\n const fullDataUri = await screenshot.domToPng(element, {\n scale: captureScale,\n backgroundColor,\n })\n\n // Resize to target dimensions (high-quality downscaling)\n const resizedDataUri = await resizeImage(\n fullDataUri,\n targetWidth,\n targetHeight,\n format,\n quality\n )\n\n return resizedDataUri\n } catch (error) {\n console.error('[drizzle-cube] Failed to capture thumbnail:', error)\n return null\n }\n}\n\n/**\n * Check if thumbnail capture is available and enabled\n */\nexport async function isThumbnailCaptureAvailable(\n config: ThumbnailFeatureConfig | undefined\n): Promise<boolean> {\n if (!config?.enabled) {\n return false\n }\n\n if (typeof window === 'undefined') {\n return false\n }\n\n const screenshot = await getScreenshotModule()\n return screenshot !== null\n}\n\n/**\n * Check if portlet screenshot-to-clipboard is available.\n * Requires both modern-screenshot AND Clipboard API with image support.\n */\nexport async function isPortletCopyAvailable(): Promise<boolean> {\n // Check if we're in browser environment\n if (typeof window === 'undefined') {\n return false\n }\n\n // Check modern-screenshot is installed\n const screenshot = await getScreenshotModule()\n if (!screenshot) {\n return false\n }\n\n // Check Clipboard API supports writing images (ClipboardItem with image/png blob)\n return (\n typeof ClipboardItem !== 'undefined' &&\n typeof navigator?.clipboard?.write === 'function'\n )\n}\n\n/**\n * Check if a background color is transparent or effectively invisible\n */\nfunction isTransparentBackground(color: string): boolean {\n if (!color || color === 'transparent' || color === 'rgba(0, 0, 0, 0)') {\n return true\n }\n // Check for rgba with 0 alpha\n const rgbaMatch = color.match(/rgba\\(\\s*\\d+\\s*,\\s*\\d+\\s*,\\s*\\d+\\s*,\\s*([\\d.]+)\\s*\\)/)\n if (rgbaMatch && parseFloat(rgbaMatch[1]) === 0) {\n return true\n }\n return false\n}\n\n/**\n * Find the effective background color by walking up the DOM tree\n */\nfunction getEffectiveBackgroundColor(element: HTMLElement): string {\n let current: HTMLElement | null = element\n\n while (current) {\n const bg = getComputedStyle(current).backgroundColor\n if (!isTransparentBackground(bg)) {\n return bg\n }\n current = current.parentElement\n }\n\n // Fallback to theme variable or white\n const themeColor = getComputedStyle(document.documentElement)\n .getPropertyValue('--dc-surface')\n .trim()\n\n return themeColor || '#ffffff'\n}\n\n/**\n * Capture a portlet element and copy to clipboard as PNG.\n * Returns true on success, false on failure.\n *\n * @param element - The HTML element to capture\n * @returns Promise<boolean> - true if successfully copied to clipboard\n */\nexport async function copyPortletToClipboard(element: HTMLElement): Promise<boolean> {\n try {\n const screenshot = await getScreenshotModule()\n if (!screenshot) {\n console.warn('[drizzle-cube] Cannot copy portlet: modern-screenshot not available')\n return false\n }\n\n // Get theme-aware background color by walking up the DOM tree\n const backgroundColor = getEffectiveBackgroundColor(element)\n\n // Capture as PNG data URL at 2x scale for retina quality\n const dataUrl = await screenshot.domToPng(element, {\n scale: 2,\n backgroundColor,\n })\n\n // Convert data URL to blob\n const response = await fetch(dataUrl)\n const blob = await response.blob()\n\n // Write to clipboard\n await navigator.clipboard.write([new ClipboardItem({ 'image/png': blob })])\n\n return true\n } catch (error) {\n console.warn('[drizzle-cube] Failed to copy portlet to clipboard:', error)\n return false\n }\n}\n","/**\n * CubeFeaturesProvider - Feature Flags Context Layer\n *\n * Provides feature configuration isolated from API and metadata layers.\n * This prevents components from re-rendering when feature flags change\n * if they don't use features.\n */\n\nimport { createContext, useContext, useState, useMemo, useCallback, useEffect, type ReactNode } from 'react'\nimport type { FeaturesConfig, DashboardLayoutMode } from '../types'\nimport { warnIfScreenshotLibMissing } from '../utils/thumbnail'\n\ninterface CubeFeaturesContextValue {\n features: FeaturesConfig\n dashboardModes: DashboardLayoutMode[]\n updateFeatures: (newFeatures: Partial<FeaturesConfig>) => void\n}\n\nconst CubeFeaturesContext = createContext<CubeFeaturesContextValue | null>(null)\n\ninterface CubeFeaturesProviderProps {\n features?: FeaturesConfig\n dashboardModes?: DashboardLayoutMode[]\n children: ReactNode\n}\n\nconst DEFAULT_FEATURES: FeaturesConfig = {\n enableAI: true,\n aiEndpoint: '/api/ai/generate',\n showSchemaDiagram: false,\n useAnalysisBuilder: false,\n editToolbar: 'both',\n floatingToolbarPosition: 'right'\n}\n\nexport function CubeFeaturesProvider({\n features: initialFeatures,\n dashboardModes = ['rows', 'grid'],\n children\n}: CubeFeaturesProviderProps) {\n // Merge passed features with defaults so new features get default values\n const [features, setFeatures] = useState<FeaturesConfig>(() => ({\n ...DEFAULT_FEATURES,\n ...initialFeatures\n }))\n\n // Warn in development if thumbnail feature is enabled but modern-screenshot is not installed\n useEffect(() => {\n warnIfScreenshotLibMissing(features.thumbnail)\n }, [features.thumbnail])\n\n const updateFeatures = useCallback((newFeatures: Partial<FeaturesConfig>) => {\n setFeatures(prev => ({ ...prev, ...newFeatures }))\n }, [])\n\n const value = useMemo(() => ({\n features,\n dashboardModes,\n updateFeatures\n }), [features, dashboardModes, updateFeatures])\n\n return (\n <CubeFeaturesContext.Provider value={value}>\n {children}\n </CubeFeaturesContext.Provider>\n )\n}\n\n// Default context value when used outside provider (safe fallback for hooks)\nconst DEFAULT_CONTEXT: CubeFeaturesContextValue = {\n features: DEFAULT_FEATURES,\n dashboardModes: ['rows', 'grid'],\n updateFeatures: () => {\n // No-op when used outside provider\n }\n}\n\n/**\n * Hook to access cube features context.\n * Returns default values if used outside CubeFeaturesProvider (graceful fallback).\n */\nexport function useCubeFeatures() {\n const context = useContext(CubeFeaturesContext)\n // Return default context if not within provider (allows hooks like useCubeLoadQuery to work in isolation)\n return context ?? DEFAULT_CONTEXT\n}\n","/**\n * ScrollContainerContext\n *\n * Provides the scroll container element for lazy loading with IntersectionObserver.\n * This allows portlets to detect visibility relative to a custom scroll container\n * (not just the viewport) when the dashboard is embedded in a scrolling div.\n */\n\nimport { createContext, useContext } from 'react'\n\nconst ScrollContainerContext = createContext<HTMLElement | null>(null)\n\n/**\n * Provider component to wrap dashboard content with a scroll container reference.\n * Used by DashboardGrid and MobileStackedLayout to pass the detected scroll container\n * to child portlets.\n */\nexport const ScrollContainerProvider = ScrollContainerContext.Provider\n\n/**\n * Hook to access the scroll container element for lazy loading.\n * Returns null if using viewport (window) scroll, or the container element\n * if the dashboard is inside a scrolling container.\n */\nexport const useScrollContainer = () => useContext(ScrollContainerContext)\n","/**\n * CubeMetaContext - Metadata context definitions.\n *\n * Split from CubeMetaProvider so consumers can access metadata context\n * without pulling in TanStack Query dependencies.\n */\n\nimport { createContext, useContext } from 'react'\nimport type { CubeMeta, FieldLabelMap } from '../types'\n\nexport interface CubeMetaContextValue {\n meta: CubeMeta | null\n labelMap: FieldLabelMap\n metaLoading: boolean\n metaError: string | null\n getFieldLabel: (fieldName: string) => string\n refetchMeta: () => void\n}\n\nexport const CubeMetaContext = createContext<CubeMetaContextValue | null>(null)\n\nexport function useCubeMeta() {\n const context = useContext(CubeMetaContext)\n if (!context) {\n throw new Error('useCubeMeta must be used within CubeMetaProvider')\n }\n return context\n}\n","/**\n * CubeMetaProvider - Metadata Context Layer\n *\n * Provides cube metadata (meta, labelMap) isolated from API layer.\n * This prevents components from re-rendering when metadata loads\n * if they only need API access.\n *\n * Uses TanStack Query for metadata fetching with built-in caching.\n */\n\nimport { useCallback, useMemo, type ReactNode } from 'react'\nimport { useCubeMetaQuery } from '../hooks/queries/useCubeMetaQuery'\nimport { CubeMetaContext } from './CubeMetaContext'\n\ninterface CubeMetaProviderProps {\n children: ReactNode\n}\n\nexport function CubeMetaProvider({ children }: CubeMetaProviderProps) {\n // Use TanStack Query hook for metadata fetching\n const {\n meta,\n labelMap,\n isLoading: metaLoading,\n error,\n refetch,\n getFieldLabel: getFieldLabelFromQuery\n } = useCubeMetaQuery()\n\n // Convert error to string for backward compatibility\n const metaError = error ? (error instanceof Error ? error.message : String(error)) : null\n\n // Wrap refetch to match expected signature\n const refetchMeta = useCallback(() => {\n refetch()\n }, [refetch])\n\n // Create stable getFieldLabel that reads from current labelMap\n const getFieldLabel = useCallback((fieldName: string): string => {\n return getFieldLabelFromQuery(fieldName)\n }, [getFieldLabelFromQuery])\n\n // Memoize context value to prevent unnecessary re-renders\n const value = useMemo(() => ({\n meta,\n labelMap,\n metaLoading,\n metaError,\n getFieldLabel,\n refetchMeta\n }), [meta, labelMap, metaLoading, metaError, getFieldLabel, refetchMeta])\n\n return (\n <CubeMetaContext.Provider value={value}>\n {children}\n </CubeMetaContext.Provider>\n )\n}\n","/**\n * CubeProvider - Backward Compatibility Wrapper\n *\n * This provider wraps the three specialized context providers (API, Meta, Features)\n * to maintain 100% backward compatibility with existing code using useCubeContext().\n *\n * New code should use the specialized hooks (useCubeApi, useCubeMeta, useCubeFeatures)\n * for better performance and selective re-rendering.\n *\n * Also includes TanStack Query's QueryClientProvider for data fetching in\n * AnalysisBuilder and other components that use TanStack Query hooks.\n */\n\nimport { useMemo, useState, type ReactNode } from 'react'\nimport { QueryClient, QueryClientProvider } from '@tanstack/react-query'\nimport { CubeApiProvider, useCubeApi } from './CubeApiProvider'\nimport { CubeMetaProvider } from './CubeMetaProvider'\nimport { useCubeMeta } from './CubeMetaContext'\nimport { CubeFeaturesProvider, useCubeFeatures } from './CubeFeaturesProvider'\nimport type { CubeQueryOptions, CubeApiOptions, FeaturesConfig, DashboardLayoutMode, CubeMeta, FieldLabelMap } from '../types'\nimport type { CubeClient } from '../client/CubeClient'\nimport type { BatchCoordinator } from '../client/BatchCoordinator'\n\nconst DEFAULT_API_OPTIONS: CubeApiOptions = { apiUrl: '/cubejs-api/v1' }\n\nexport const createCubeQueryClient = () => new QueryClient({\n defaultOptions: {\n queries: {\n // Stale time: 5 minutes (matches existing cache duration)\n staleTime: 5 * 60 * 1000,\n // GC time: 15 minutes\n gcTime: 15 * 60 * 1000,\n // Retry failed queries up to 3 times\n retry: 3,\n // Don't refetch on window focus by default (can be overridden per-query)\n refetchOnWindowFocus: false,\n },\n },\n})\n\n// Legacy default client (useful for tests or explicit injection)\nexport const queryClient = createCubeQueryClient()\n\n// Backward compatible interface - merges all three contexts\ninterface CubeContextValue {\n cubeApi: CubeClient\n options?: CubeQueryOptions\n meta: CubeMeta | null\n labelMap: FieldLabelMap\n metaLoading: boolean\n metaError: string | null\n getFieldLabel: (fieldName: string) => string\n refetchMeta: () => void\n updateApiConfig: (apiOptions: CubeApiOptions, token?: string) => void\n features: FeaturesConfig\n batchCoordinator: BatchCoordinator | null\n enableBatching: boolean\n dashboardModes: DashboardLayoutMode[]\n}\n\ninterface CubeProviderProps {\n cubeApi?: CubeClient\n apiOptions?: CubeApiOptions\n token?: string\n options?: CubeQueryOptions\n features?: FeaturesConfig\n dashboardModes?: DashboardLayoutMode[]\n enableBatching?: boolean\n batchDelayMs?: number // Delay in ms to collect queries before batching (default: 100)\n queryClient?: QueryClient\n children: ReactNode\n}\n\n/**\n * CubeProvider - Three-layer context wrapper\n *\n * Wraps children in three isolated context providers for optimal performance:\n * 1. CubeApiProvider - Stable API layer (changes only on auth)\n * 2. CubeMetaProvider - Metadata layer (changes on metadata load)\n * 3. CubeFeaturesProvider - Feature flags layer (changes on feature updates)\n */\nexport function CubeProvider({\n cubeApi: _initialCubeApi, // Intentionally unused - for backward compatibility\n apiOptions,\n token,\n options,\n features,\n dashboardModes,\n enableBatching,\n batchDelayMs,\n queryClient: providedQueryClient,\n children\n}: CubeProviderProps) {\n const [internalQueryClient] = useState(() => createCubeQueryClient())\n const queryClient = providedQueryClient ?? internalQueryClient\n\n return (\n <QueryClientProvider client={queryClient}>\n <CubeApiProvider\n apiOptions={apiOptions || DEFAULT_API_OPTIONS}\n token={token}\n options={options}\n enableBatching={enableBatching}\n batchDelayMs={batchDelayMs}\n >\n <CubeMetaProvider>\n <CubeFeaturesProvider features={features} dashboardModes={dashboardModes}>\n {children}\n </CubeFeaturesProvider>\n </CubeMetaProvider>\n </CubeApiProvider>\n </QueryClientProvider>\n )\n}\n\n/**\n * useCubeContext - Backward compatible hook\n *\n * Merges all three contexts into a single object for backward compatibility.\n * Components using this hook will re-render when ANY context changes.\n *\n * For better performance, use specialized hooks:\n * - useCubeApi() - Only re-renders on API changes\n * - useCubeMeta() - Only re-renders on metadata changes\n * - useCubeFeatures() - Only re-renders on feature changes\n */\nexport function useCubeContext(): CubeContextValue {\n const api = useCubeApi()\n const meta = useCubeMeta()\n const featuresCtx = useCubeFeatures()\n\n return useMemo(() => ({\n ...api,\n ...meta,\n features: featuresCtx.features,\n dashboardModes: featuresCtx.dashboardModes\n }), [api, meta, featuresCtx])\n}\n\n// Re-export specialized hooks for better tree-shaking and performance\nexport { useCubeApi, useCubeMeta, useCubeFeatures }\n\n// Export factory for testing and advanced use cases\nexport { createCubeQueryClient as createQueryClient }\n"],"names":["CubeClient","token","options","query","queryString","queryParam","url","requestHeaders","key","response","errorMessage","errorText","errorData","result","ResultSet","queries","loadResponse","createCubeClient","BatchCoordinator","batchExecutor","delayMs","resolve","reject","currentQueue","item","results","index","error","CubeApiContext","createContext","CubeApiProvider","initialApiOptions","initialToken","enableBatching","batchDelayMs","children","baseConfig","useMemo","overrideConfig","setOverrideConfig","useState","useEffect","config","cubeApi","batchCoordinator","updateApiConfig","useCallback","newApiOptions","newToken","value","jsx","useCubeApi","context","useContext","CUBE_META_QUERY_KEY","buildLabelMap","meta","labelMap","cube","measure","dimension","segment","useCubeMetaQuery","enabled","staleTime","queryClient","useQueryClient","useQuery","metaData","refetch","getFieldLabel","fieldName","screenshotModule","moduleChecked","resizeImage","dataUri","targetWidth","targetHeight","format","quality","img","canvas","ctx","sourceAspect","targetAspect","sourceX","sourceY","sourceWidth","sourceHeight","mimeType","getScreenshotModule","warnIfScreenshotLibMissing","thumbnailConfig","mod","captureThumbnail","elementRef","screenshot","element","captureScale","backgroundColor","getEffectiveBackgroundColor","fullDataUri","isPortletCopyAvailable","isTransparentBackground","color","rgbaMatch","current","bg","copyPortletToClipboard","dataUrl","blob","CubeFeaturesContext","DEFAULT_FEATURES","CubeFeaturesProvider","initialFeatures","dashboardModes","features","setFeatures","updateFeatures","newFeatures","prev","DEFAULT_CONTEXT","useCubeFeatures","ScrollContainerContext","ScrollContainerProvider","useScrollContainer","CubeMetaContext","useCubeMeta","CubeMetaProvider","metaLoading","getFieldLabelFromQuery","metaError","refetchMeta","DEFAULT_API_OPTIONS","createCubeQueryClient","QueryClient","CubeProvider","_initialCubeApi","apiOptions","providedQueryClient","internalQueryClient","QueryClientProvider","useCubeContext","api","featuresCtx"],"mappings":"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"}
@@ -0,0 +1,120 @@
1
+ const o = [
2
+ { value: "last_30_days", label: "Last 30 days" },
3
+ { value: "last_3_months", label: "Last 3 months" },
4
+ { value: "last_6_months", label: "Last 6 months" },
5
+ { value: "last_12_months", label: "Last 12 months" },
6
+ { value: "this_year", label: "This year" },
7
+ { value: "last_year", label: "Last year" },
8
+ { value: "custom", label: "Custom range" }
9
+ ], i = "last_3_months";
10
+ function l(e) {
11
+ const a = /* @__PURE__ */ new Date(), t = new Date(a.getFullYear(), a.getMonth(), a.getDate());
12
+ switch (e) {
13
+ case "last_30_days": {
14
+ const n = new Date(t);
15
+ return n.setDate(n.getDate() - 30), {
16
+ start: s(n),
17
+ end: s(t)
18
+ };
19
+ }
20
+ case "last_3_months": {
21
+ const n = new Date(t.getFullYear(), t.getMonth() - 3, 1), r = new Date(t.getFullYear(), t.getMonth(), 0);
22
+ return {
23
+ start: s(n),
24
+ end: s(r)
25
+ };
26
+ }
27
+ case "last_6_months": {
28
+ const n = new Date(t.getFullYear(), t.getMonth() - 6, 1), r = new Date(t.getFullYear(), t.getMonth(), 0);
29
+ return {
30
+ start: s(n),
31
+ end: s(r)
32
+ };
33
+ }
34
+ case "last_12_months": {
35
+ const n = new Date(t.getFullYear(), t.getMonth() - 12, 1), r = new Date(t.getFullYear(), t.getMonth(), 0);
36
+ return {
37
+ start: s(n),
38
+ end: s(r)
39
+ };
40
+ }
41
+ case "this_year": {
42
+ const n = new Date(t.getFullYear(), 0, 1);
43
+ return {
44
+ start: s(n),
45
+ end: s(t)
46
+ };
47
+ }
48
+ case "last_year": {
49
+ const n = new Date(t.getFullYear() - 1, 0, 1), r = new Date(t.getFullYear() - 1, 11, 31);
50
+ return {
51
+ start: s(n),
52
+ end: s(r)
53
+ };
54
+ }
55
+ default:
56
+ return l("last_3_months");
57
+ }
58
+ }
59
+ function s(e) {
60
+ const a = e.getFullYear(), t = String(e.getMonth() + 1).padStart(2, "0"), n = String(e.getDate()).padStart(2, "0");
61
+ return `${a}-${t}-${n}`;
62
+ }
63
+ function c(e) {
64
+ for (const a of o) {
65
+ if (a.value === "custom") continue;
66
+ const t = l(a.value);
67
+ if (t.start === e.start && t.end === e.end)
68
+ return a.value;
69
+ }
70
+ return "custom";
71
+ }
72
+ function u(e) {
73
+ if (!e || typeof e != "object") return !1;
74
+ const a = e;
75
+ return Array.isArray(a.rows) && Array.isArray(a.periods);
76
+ }
77
+ function _(e) {
78
+ return typeof e == "object" && e !== null && "retention" in e && typeof e.retention == "object";
79
+ }
80
+ const g = {
81
+ retentionCube: null,
82
+ retentionBindingKey: null,
83
+ retentionTimeDimension: null,
84
+ retentionDateRange: l(i),
85
+ retentionCohortFilters: [],
86
+ retentionActivityFilters: [],
87
+ retentionBreakdowns: [],
88
+ retentionViewGranularity: "week",
89
+ retentionPeriods: 12,
90
+ retentionType: "classic"
91
+ }, d = 1, y = 52, D = [
92
+ { value: "day", label: "Daily" },
93
+ { value: "week", label: "Weekly" },
94
+ { value: "month", label: "Monthly" }
95
+ ], h = [
96
+ {
97
+ value: "classic",
98
+ label: "Classic",
99
+ description: "User was active in exactly period N"
100
+ },
101
+ {
102
+ value: "rolling",
103
+ label: "Rolling",
104
+ description: "User was active in period N or later"
105
+ }
106
+ ];
107
+ export {
108
+ i as D,
109
+ d as R,
110
+ y as a,
111
+ c as b,
112
+ o as c,
113
+ g as d,
114
+ D as e,
115
+ h as f,
116
+ l as g,
117
+ _ as h,
118
+ u as i
119
+ };
120
+ //# sourceMappingURL=retention-CzCo8262.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"retention-CzCo8262.js","sources":["../../../src/client/types/retention.ts"],"sourcesContent":["/**\n * Retention Analysis Types\n *\n * Types for Retention analysis mode which tracks cohort-based retention over time.\n * Retention analysis measures what percentage of users from each cohort return\n * in subsequent time periods.\n */\n\nimport type { Filter } from '../types'\nimport type { FunnelBindingKey } from './funnel'\n\n// ============================================================================\n// Date Range Types\n// ============================================================================\n\n/**\n * Date range for cohort analysis (REQUIRED)\n * Matches server's RetentionDateRange interface\n */\nexport interface DateRange {\n /** Start date (inclusive), ISO 8601 format (YYYY-MM-DD) */\n start: string\n /** End date (inclusive), ISO 8601 format (YYYY-MM-DD) */\n end: string\n}\n\n/**\n * Preset date range type\n */\nexport type DateRangePreset =\n | 'last_30_days'\n | 'last_3_months'\n | 'last_6_months'\n | 'last_12_months'\n | 'this_year'\n | 'last_year'\n | 'custom'\n\n/**\n * Preset date range options for the UI\n */\nexport const RETENTION_DATE_RANGE_PRESETS: { value: DateRangePreset; label: string }[] = [\n { value: 'last_30_days', label: 'Last 30 days' },\n { value: 'last_3_months', label: 'Last 3 months' },\n { value: 'last_6_months', label: 'Last 6 months' },\n { value: 'last_12_months', label: 'Last 12 months' },\n { value: 'this_year', label: 'This year' },\n { value: 'last_year', label: 'Last year' },\n { value: 'custom', label: 'Custom range' },\n]\n\n/**\n * Default preset for retention date range\n */\nexport const DEFAULT_DATE_RANGE_PRESET: DateRangePreset = 'last_3_months'\n\n/**\n * Calculate date range from a preset value\n * Returns ISO date strings (YYYY-MM-DD)\n */\nexport function getDateRangeFromPreset(preset: DateRangePreset): DateRange {\n const now = new Date()\n const today = new Date(now.getFullYear(), now.getMonth(), now.getDate())\n\n switch (preset) {\n case 'last_30_days': {\n const start = new Date(today)\n start.setDate(start.getDate() - 30)\n return {\n start: formatDateToISO(start),\n end: formatDateToISO(today),\n }\n }\n case 'last_3_months': {\n // Start of 3 months ago to end of last month\n const start = new Date(today.getFullYear(), today.getMonth() - 3, 1)\n const end = new Date(today.getFullYear(), today.getMonth(), 0) // Last day of previous month\n return {\n start: formatDateToISO(start),\n end: formatDateToISO(end),\n }\n }\n case 'last_6_months': {\n const start = new Date(today.getFullYear(), today.getMonth() - 6, 1)\n const end = new Date(today.getFullYear(), today.getMonth(), 0)\n return {\n start: formatDateToISO(start),\n end: formatDateToISO(end),\n }\n }\n case 'last_12_months': {\n const start = new Date(today.getFullYear(), today.getMonth() - 12, 1)\n const end = new Date(today.getFullYear(), today.getMonth(), 0)\n return {\n start: formatDateToISO(start),\n end: formatDateToISO(end),\n }\n }\n case 'this_year': {\n const start = new Date(today.getFullYear(), 0, 1)\n return {\n start: formatDateToISO(start),\n end: formatDateToISO(today),\n }\n }\n case 'last_year': {\n const start = new Date(today.getFullYear() - 1, 0, 1)\n const end = new Date(today.getFullYear() - 1, 11, 31)\n return {\n start: formatDateToISO(start),\n end: formatDateToISO(end),\n }\n }\n case 'custom':\n default:\n // For custom, return last 3 months as fallback\n return getDateRangeFromPreset('last_3_months')\n }\n}\n\n/**\n * Format a Date object to ISO date string (YYYY-MM-DD)\n */\nfunction formatDateToISO(date: Date): string {\n const year = date.getFullYear()\n const month = String(date.getMonth() + 1).padStart(2, '0')\n const day = String(date.getDate()).padStart(2, '0')\n return `${year}-${month}-${day}`\n}\n\n/**\n * Detect which preset a date range matches, if any\n */\nexport function detectDateRangePreset(range: DateRange): DateRangePreset {\n for (const preset of RETENTION_DATE_RANGE_PRESETS) {\n if (preset.value === 'custom') continue\n const presetRange = getDateRangeFromPreset(preset.value)\n if (presetRange.start === range.start && presetRange.end === range.end) {\n return preset.value\n }\n }\n return 'custom'\n}\n\n// ============================================================================\n// Server Query Format\n// ============================================================================\n\n/**\n * Server retention query format\n * This is the shape sent to the server for execution\n * Wrapped in { retention: {...} } similar to funnel queries\n */\nexport interface ServerRetentionQuery {\n retention: RetentionQueryConfig\n}\n\n/**\n * Retention query configuration\n * Contains all parameters needed for server-side retention analysis\n *\n * Simplified Mixpanel-style format:\n * - Single cube and timestamp dimension\n * - Single cohort (date range defines the cohort, not granularity)\n * - Optional breakdown dimension for segmentation\n */\nexport interface RetentionQueryConfig {\n /**\n * Single timestamp dimension for the analysis.\n * String format (e.g., 'Events.timestamp'),\n * Object format for multi-cube with explicit cube reference.\n */\n timeDimension: string | { cube: string; dimension: string }\n\n /**\n * Binding key - dimension that links users across events.\n * This is typically a user ID or other entity identifier.\n * String for single-cube (e.g., 'Events.userId'),\n * Array for multi-cube with different column names per cube.\n */\n bindingKey: string | { cube: string; dimension: string }[]\n\n /**\n * Date range for cohort analysis (REQUIRED).\n * Users who first performed the cohort action within this range are included.\n */\n dateRange: DateRange\n\n /**\n * Granularity for viewing retention periods.\n * Determines how retention periods are measured (day/week/month).\n */\n granularity: RetentionGranularity\n\n /**\n * Number of periods to calculate (e.g., 12 for 12 weeks).\n * Period 0 is always the cohort entry period.\n */\n periods: number\n\n /**\n * Retention type:\n * - 'classic': User returned exactly in period N (bounded)\n * - 'rolling': User returned in period N or any later period (unbounded)\n */\n retentionType: RetentionType\n\n /**\n * Optional filters on cohort entry events.\n * Applied when identifying which users enter the cohort.\n */\n cohortFilters?: Filter | Filter[]\n\n /**\n * Optional filters on return activity events.\n * Applied when checking for user activity in each period.\n */\n activityFilters?: Filter | Filter[]\n\n /**\n * Optional breakdown dimensions for segmenting the cohort.\n * When provided, retention is calculated per breakdown value combination.\n * e.g., [\"Events.country\"] or [\"Events.country\", \"Events.plan\"]\n */\n breakdownDimensions?: string[]\n}\n\n// ============================================================================\n// Enums and Unions\n// ============================================================================\n\n/**\n * Supported granularity levels for retention analysis\n */\nexport type RetentionGranularity = 'day' | 'week' | 'month'\n\n/**\n * Retention calculation types\n * - classic: User active exactly in period N\n * - rolling: User active in period N or any later period\n */\nexport type RetentionType = 'classic' | 'rolling'\n\n// ============================================================================\n// Result Types\n// ============================================================================\n\n/**\n * Single retention data point returned from server\n * Results are returned as a flat array; client transforms to matrix if needed\n */\nexport interface RetentionResultRow {\n /** Period number (0 = cohort entry, 1 = first retention period, etc.) */\n period: number\n\n /** Number of users in the cohort */\n cohortSize: number\n\n /** Number of users retained in this period */\n retainedUsers: number\n\n /** Retention rate as decimal (0-1), e.g., 0.45 for 45% */\n retentionRate: number\n\n /** Breakdown value when breakdown dimension is specified (e.g., \"US\", \"UK\") */\n breakdownValue?: string | null\n}\n\n/**\n * Retention chart data format for visualization\n * Supports both heatmap and line chart modes\n */\nexport interface RetentionChartData {\n rows: RetentionResultRow[]\n /** Period numbers (0 to periods) */\n periods: number[]\n /** Breakdown values when breakdown dimension is specified */\n breakdownValues?: string[]\n /** Summary statistics */\n summary?: RetentionSummary\n /** Granularity of retention periods (day/week/month) for period label formatting */\n granularity?: RetentionGranularity\n /** Human-readable label extracted from the binding key dimension (e.g., \"userId\" from \"Users.userId\") */\n bindingKeyLabel?: string\n}\n\n/**\n * Summary statistics for retention analysis\n */\nexport interface RetentionSummary {\n /** Total unique users in the cohort */\n totalUsers: number\n /** Average retention rate across all periods for period 1 */\n avgPeriod1Retention: number\n /** Highest retention rate for period 1 */\n maxPeriod1Retention: number\n /** Lowest retention rate for period 1 */\n minPeriod1Retention: number\n /** Number of breakdown segments (1 if no breakdown) */\n segmentCount?: number\n}\n\n// ============================================================================\n// Breakdown Types\n// ============================================================================\n\n/**\n * Breakdown item for retention analysis (single dimension)\n * Follows Mixpanel pattern - one breakdown dimension only\n */\nexport interface RetentionBreakdownItem {\n /** Full dimension name (e.g., \"Events.country\") */\n field: string\n /** Display label for the dimension */\n label?: string\n}\n\n// ============================================================================\n// Slice State (for Zustand store)\n// ============================================================================\n\n/**\n * Retention mode state for the AnalysisBuilder store\n * Simplified Mixpanel-style with single global configuration\n *\n * Key simplifications from previous version:\n * - Single cube for all (no separate cohort/activity cubes)\n * - Single timestamp dimension\n * - Single cohort with breakdown support (no cohort explosion)\n * - Granularity = viewing periods only\n */\nexport interface RetentionSliceState {\n /** Single cube for retention analysis */\n retentionCube: string | null\n\n /** Binding key that identifies entities (reuses funnel binding key type) */\n retentionBindingKey: FunnelBindingKey | null\n\n /** Single timestamp dimension for both cohort entry and activity */\n retentionTimeDimension: string | null\n\n /** Date range for cohort analysis (REQUIRED) */\n retentionDateRange: DateRange\n\n /** Filters that define who enters the cohort */\n retentionCohortFilters: Filter[]\n\n /** Filters that define what counts as a return */\n retentionActivityFilters: Filter[]\n\n /** Optional breakdown dimensions for segmenting the cohort */\n retentionBreakdowns: RetentionBreakdownItem[]\n\n /** Granularity for viewing retention periods (day/week/month) */\n retentionViewGranularity: RetentionGranularity\n\n /** Number of periods to analyze (1-52) */\n retentionPeriods: number\n\n /** Type of retention calculation */\n retentionType: RetentionType\n}\n\n/**\n * Retention slice actions for the store\n */\nexport interface RetentionSliceActions {\n /** Set the single cube for retention analysis */\n setRetentionCube: (cube: string | null) => void\n /** Set the retention binding key */\n setRetentionBindingKey: (key: FunnelBindingKey | null) => void\n /** Set the single timestamp dimension */\n setRetentionTimeDimension: (dim: string | null) => void\n /** Set the date range (REQUIRED) */\n setRetentionDateRange: (range: DateRange) => void\n /** Set all cohort filters at once */\n setRetentionCohortFilters: (filters: Filter[]) => void\n /** Add a cohort filter */\n addRetentionCohortFilter: (filter: Filter) => void\n /** Remove a cohort filter by index */\n removeRetentionCohortFilter: (index: number) => void\n /** Update a cohort filter by index */\n updateRetentionCohortFilter: (index: number, filter: Filter) => void\n /** Set all activity filters at once */\n setRetentionActivityFilters: (filters: Filter[]) => void\n /** Add an activity filter */\n addRetentionActivityFilter: (filter: Filter) => void\n /** Remove an activity filter by index */\n removeRetentionActivityFilter: (index: number) => void\n /** Update an activity filter by index */\n updateRetentionActivityFilter: (index: number, filter: Filter) => void\n /** Set all breakdown dimensions */\n setRetentionBreakdowns: (breakdowns: RetentionBreakdownItem[]) => void\n /** Add a breakdown dimension */\n addRetentionBreakdown: (breakdown: RetentionBreakdownItem) => void\n /** Remove a breakdown dimension by field name */\n removeRetentionBreakdown: (field: string) => void\n /** Set the view granularity */\n setRetentionViewGranularity: (granularity: RetentionGranularity) => void\n /** Set the number of periods */\n setRetentionPeriods: (periods: number) => void\n /** Set the retention type */\n setRetentionType: (type: RetentionType) => void\n /** Check if in retention mode (analysisType === 'retention') */\n isRetentionMode: () => boolean\n /** Check if retention mode is properly configured and ready for execution */\n isRetentionModeEnabled: () => boolean\n /** Build ServerRetentionQuery from retention state */\n buildRetentionQuery: () => ServerRetentionQuery | null\n /** Get validation errors explaining why retention query cannot be built */\n getRetentionValidation: () => { isValid: boolean; errors: string[]; warnings: string[] }\n}\n\n// ============================================================================\n// Type Guards\n// ============================================================================\n\n/**\n * Type guard to check if data is retention chart data\n */\nexport function isRetentionData(data: unknown): data is RetentionChartData {\n if (!data || typeof data !== 'object') return false\n const d = data as Record<string, unknown>\n return (\n Array.isArray(d.rows) &&\n Array.isArray(d.periods)\n )\n}\n\n/**\n * Type guard to detect server retention query format\n * Used to distinguish { retention: {...} } from CubeQuery, etc.\n */\nexport function isServerRetentionQuery(obj: unknown): obj is ServerRetentionQuery {\n return (\n typeof obj === 'object' &&\n obj !== null &&\n 'retention' in obj &&\n typeof (obj as { retention: unknown }).retention === 'object'\n )\n}\n\n/**\n * Type guard for retention result row\n */\nexport function isRetentionResultRow(row: unknown): row is RetentionResultRow {\n if (!row || typeof row !== 'object') return false\n const r = row as Record<string, unknown>\n return (\n typeof r.cohortPeriod === 'string' &&\n typeof r.period === 'number' &&\n typeof r.cohortSize === 'number' &&\n typeof r.retainedUsers === 'number' &&\n typeof r.retentionRate === 'number'\n )\n}\n\n// ============================================================================\n// Default Values\n// ============================================================================\n\n/**\n * Default retention slice state for store initialization\n */\nexport const defaultRetentionSliceState: RetentionSliceState = {\n retentionCube: null,\n retentionBindingKey: null,\n retentionTimeDimension: null,\n retentionDateRange: getDateRangeFromPreset(DEFAULT_DATE_RANGE_PRESET),\n retentionCohortFilters: [],\n retentionActivityFilters: [],\n retentionBreakdowns: [],\n retentionViewGranularity: 'week',\n retentionPeriods: 12,\n retentionType: 'classic',\n}\n\n/**\n * Minimum and maximum values for retention periods\n */\nexport const RETENTION_MIN_PERIODS = 1\nexport const RETENTION_MAX_PERIODS = 52\n\n/**\n * Available granularity options\n */\nexport const RETENTION_GRANULARITY_OPTIONS: { value: RetentionGranularity; label: string }[] = [\n { value: 'day', label: 'Daily' },\n { value: 'week', label: 'Weekly' },\n { value: 'month', label: 'Monthly' },\n]\n\n/**\n * Available retention type options\n */\nexport const RETENTION_TYPE_OPTIONS: { value: RetentionType; label: string; description: string }[] = [\n {\n value: 'classic',\n label: 'Classic',\n description: 'User was active in exactly period N',\n },\n {\n value: 'rolling',\n label: 'Rolling',\n description: 'User was active in period N or later',\n },\n]\n"],"names":["RETENTION_DATE_RANGE_PRESETS","DEFAULT_DATE_RANGE_PRESET","getDateRangeFromPreset","preset","now","today","start","formatDateToISO","end","date","year","month","day","detectDateRangePreset","range","presetRange","isRetentionData","data","d","isServerRetentionQuery","obj","defaultRetentionSliceState","RETENTION_MIN_PERIODS","RETENTION_MAX_PERIODS","RETENTION_GRANULARITY_OPTIONS","RETENTION_TYPE_OPTIONS"],"mappings":"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"}