drizzle-cube 0.4.21 → 0.4.23

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (102) hide show
  1. package/dist/adapters/express/index.cjs +1 -1
  2. package/dist/adapters/express/index.js +2 -2
  3. package/dist/adapters/fastify/index.cjs +1 -1
  4. package/dist/adapters/fastify/index.js +2 -2
  5. package/dist/adapters/{handler-B-tEntiU.cjs → handler-C3hT7g2W.cjs} +1 -1
  6. package/dist/adapters/{handler-9Rdn7zM2.js → handler-t7Qd1IYi.js} +1 -1
  7. package/dist/adapters/hono/index.cjs +6 -6
  8. package/dist/adapters/hono/index.d.ts +13 -6
  9. package/dist/adapters/hono/index.js +65 -65
  10. package/dist/adapters/{mcp-transport-m1X1GtwG.js → mcp-transport-B6ZudTSk.js} +7 -0
  11. package/dist/adapters/{mcp-transport-8u9G5oNa.cjs → mcp-transport-DCiSGtp1.cjs} +1 -1
  12. package/dist/adapters/nextjs/index.cjs +2 -2
  13. package/dist/adapters/nextjs/index.js +2 -2
  14. package/dist/client/charts.js +12 -12
  15. package/dist/client/chunks/{RetentionCombinedChart-CLq89aOJ.js → RetentionCombinedChart-BK3NPsHP.js} +2 -2
  16. package/dist/client/chunks/{RetentionCombinedChart-CLq89aOJ.js.map → RetentionCombinedChart-BK3NPsHP.js.map} +1 -1
  17. package/dist/client/chunks/{analysis-builder-3z9fHE2F.js → analysis-builder-DVrv9Q4n.js} +9 -9
  18. package/dist/client/chunks/{analysis-builder-3z9fHE2F.js.map → analysis-builder-DVrv9Q4n.js.map} +1 -1
  19. package/dist/client/chunks/{analysis-builder-shared-Da-vlQa_.js → analysis-builder-shared-CrENEvEk.js} +6 -6
  20. package/dist/client/chunks/{analysis-builder-shared-Da-vlQa_.js.map → analysis-builder-shared-CrENEvEk.js.map} +1 -1
  21. package/dist/client/chunks/{chart-activity-grid-CPGcTSuh.js → chart-activity-grid-OG6he4YS.js} +2 -2
  22. package/dist/client/chunks/{chart-activity-grid-CPGcTSuh.js.map → chart-activity-grid-OG6he4YS.js.map} +1 -1
  23. package/dist/client/chunks/{chart-area-ByJQ7NZd.js → chart-area-TawAd2k9.js} +3 -3
  24. package/dist/client/chunks/{chart-area-ByJQ7NZd.js.map → chart-area-TawAd2k9.js.map} +1 -1
  25. package/dist/client/chunks/{chart-bar-dj14frMt.js → chart-bar-D3vtCNQG.js} +2 -2
  26. package/dist/client/chunks/{chart-bar-dj14frMt.js.map → chart-bar-D3vtCNQG.js.map} +1 -1
  27. package/dist/client/chunks/{chart-box-plot-ZatBpatq.js → chart-box-plot-BXwN2rO5.js} +2 -2
  28. package/dist/client/chunks/{chart-box-plot-ZatBpatq.js.map → chart-box-plot-BXwN2rO5.js.map} +1 -1
  29. package/dist/client/chunks/{chart-bubble-CemotLx-.js → chart-bubble-ZfNe8t5k.js} +2 -2
  30. package/dist/client/chunks/{chart-bubble-CemotLx-.js.map → chart-bubble-ZfNe8t5k.js.map} +1 -1
  31. package/dist/client/chunks/{chart-candlestick-BIR4uGGt.js → chart-candlestick-DmF3haFu.js} +2 -2
  32. package/dist/client/chunks/{chart-candlestick-BIR4uGGt.js.map → chart-candlestick-DmF3haFu.js.map} +1 -1
  33. package/dist/client/chunks/{chart-data-table-BsAjHe7o.js → chart-data-table-DJZPkArt.js} +4 -4
  34. package/dist/client/chunks/{chart-data-table-BsAjHe7o.js.map → chart-data-table-DJZPkArt.js.map} +1 -1
  35. package/dist/client/chunks/{chart-funnel-dofnhD24.js → chart-funnel-CE9x0Io9.js} +2 -2
  36. package/dist/client/chunks/{chart-funnel-dofnhD24.js.map → chart-funnel-CE9x0Io9.js.map} +1 -1
  37. package/dist/client/chunks/{chart-gauge-CKJJ8m3b.js → chart-gauge-Djs5FWxB.js} +2 -2
  38. package/dist/client/chunks/{chart-gauge-CKJJ8m3b.js.map → chart-gauge-Djs5FWxB.js.map} +1 -1
  39. package/dist/client/chunks/{chart-heat-map-BVuPUKHT.js → chart-heat-map-CqtMkdxd.js} +2 -2
  40. package/dist/client/chunks/{chart-heat-map-BVuPUKHT.js.map → chart-heat-map-CqtMkdxd.js.map} +1 -1
  41. package/dist/client/chunks/{chart-kpi-delta-DUD3f8vL.js → chart-kpi-delta-DEzA74cL.js} +4 -4
  42. package/dist/client/chunks/{chart-kpi-delta-DUD3f8vL.js.map → chart-kpi-delta-DEzA74cL.js.map} +1 -1
  43. package/dist/client/chunks/{chart-kpi-number-iJh-PzsM.js → chart-kpi-number-Bab-BZtX.js} +3 -3
  44. package/dist/client/chunks/{chart-kpi-number-iJh-PzsM.js.map → chart-kpi-number-Bab-BZtX.js.map} +1 -1
  45. package/dist/client/chunks/{chart-kpi-text-x6pV9v9Q.js → chart-kpi-text-BkTgckDJ.js} +3 -3
  46. package/dist/client/chunks/{chart-kpi-text-x6pV9v9Q.js.map → chart-kpi-text-BkTgckDJ.js.map} +1 -1
  47. package/dist/client/chunks/{chart-line-DzyZkugh.js → chart-line-DhM-Hvu0.js} +3 -3
  48. package/dist/client/chunks/{chart-line-DzyZkugh.js.map → chart-line-DhM-Hvu0.js.map} +1 -1
  49. package/dist/client/chunks/{chart-measure-profile-C2IkBG3V.js → chart-measure-profile-4vQxFm69.js} +3 -3
  50. package/dist/client/chunks/{chart-measure-profile-C2IkBG3V.js.map → chart-measure-profile-4vQxFm69.js.map} +1 -1
  51. package/dist/client/chunks/{chart-pie-akbfRfb9.js → chart-pie-B86KRcHI.js} +2 -2
  52. package/dist/client/chunks/{chart-pie-akbfRfb9.js.map → chart-pie-B86KRcHI.js.map} +1 -1
  53. package/dist/client/chunks/{chart-radar-BaN-Kjww.js → chart-radar-BhDBmJRh.js} +2 -2
  54. package/dist/client/chunks/{chart-radar-BaN-Kjww.js.map → chart-radar-BhDBmJRh.js.map} +1 -1
  55. package/dist/client/chunks/{chart-radial-bar-DpptEL3s.js → chart-radial-bar-Brugya8X.js} +2 -2
  56. package/dist/client/chunks/{chart-radial-bar-DpptEL3s.js.map → chart-radial-bar-Brugya8X.js.map} +1 -1
  57. package/dist/client/chunks/{chart-sankey-CG-3hHmX.js → chart-sankey-D2L8ympI.js} +2 -2
  58. package/dist/client/chunks/{chart-sankey-CG-3hHmX.js.map → chart-sankey-D2L8ympI.js.map} +1 -1
  59. package/dist/client/chunks/{chart-scatter-l_yTVxF3.js → chart-scatter-CAkbBDkK.js} +2 -2
  60. package/dist/client/chunks/{chart-scatter-l_yTVxF3.js.map → chart-scatter-CAkbBDkK.js.map} +1 -1
  61. package/dist/client/chunks/{chart-sunburst-KhDcKhmZ.js → chart-sunburst-DaxJ-cob.js} +2 -2
  62. package/dist/client/chunks/{chart-sunburst-KhDcKhmZ.js.map → chart-sunburst-DaxJ-cob.js.map} +1 -1
  63. package/dist/client/chunks/{chart-tree-map-CBbiaBXV.js → chart-tree-map-CrDJAvZU.js} +2 -2
  64. package/dist/client/chunks/{chart-tree-map-CBbiaBXV.js.map → chart-tree-map-CrDJAvZU.js.map} +1 -1
  65. package/dist/client/chunks/{chart-waterfall-CX3vx_lI.js → chart-waterfall-BBwSfEKT.js} +3 -3
  66. package/dist/client/chunks/{chart-waterfall-CX3vx_lI.js.map → chart-waterfall-BBwSfEKT.js.map} +1 -1
  67. package/dist/client/chunks/{charts-core-CU9u_HtL.js → charts-core-B4Rbfdcn.js} +2 -2
  68. package/dist/client/chunks/{charts-core-CU9u_HtL.js.map → charts-core-B4Rbfdcn.js.map} +1 -1
  69. package/dist/client/chunks/{charts-loader-B3tt5oKG.js → charts-loader-DbrwgvCK.js} +25 -25
  70. package/dist/client/chunks/{charts-loader-B3tt5oKG.js.map → charts-loader-DbrwgvCK.js.map} +1 -1
  71. package/dist/client/chunks/{components-CMGGxqOB.js → components-GzooQM5J.js} +9 -9
  72. package/dist/client/chunks/{components-CMGGxqOB.js.map → components-GzooQM5J.js.map} +1 -1
  73. package/dist/client/chunks/{core-D_8mkGpQ.js → core-Y9e-sNfb.js} +2 -2
  74. package/dist/client/chunks/{core-D_8mkGpQ.js.map → core-Y9e-sNfb.js.map} +1 -1
  75. package/dist/client/chunks/{icons-M7shurcH.js → icons-DFJw-2HU.js} +6 -6
  76. package/dist/client/chunks/{icons-M7shurcH.js.map → icons-DFJw-2HU.js.map} +1 -1
  77. package/dist/client/chunks/{providers-CgxXm6Ll.js → providers-CCw8Kjlc.js} +2 -2
  78. package/dist/client/chunks/{providers-CgxXm6Ll.js.map → providers-CCw8Kjlc.js.map} +1 -1
  79. package/dist/client/chunks/{syntaxHighlighting-BQfjio-i.js → syntaxHighlighting-DAMSW_A6.js} +2 -2
  80. package/dist/client/chunks/{syntaxHighlighting-BQfjio-i.js.map → syntaxHighlighting-DAMSW_A6.js.map} +1 -1
  81. package/dist/client/chunks/{useDirtyStateTracking-Cu1HSjmo.js → useDirtyStateTracking-CjhwBXRw.js} +20 -20
  82. package/dist/client/chunks/{useDirtyStateTracking-Cu1HSjmo.js.map → useDirtyStateTracking-CjhwBXRw.js.map} +1 -1
  83. package/dist/client/chunks/useExplainAI-IiW55BaQ.js +182 -0
  84. package/dist/client/chunks/useExplainAI-IiW55BaQ.js.map +1 -0
  85. package/dist/client/chunks/{vendor-AVsJ2ni0.js → vendor-B2EH3V58.js} +7 -7
  86. package/dist/client/chunks/{vendor-AVsJ2ni0.js.map → vendor-B2EH3V58.js.map} +1 -1
  87. package/dist/client/components.js +1 -1
  88. package/dist/client/hooks/useNotebookLayout.d.ts +8 -0
  89. package/dist/client/hooks.d.ts +2 -0
  90. package/dist/client/hooks.js +51 -190
  91. package/dist/client/hooks.js.map +1 -1
  92. package/dist/client/icons.js +1 -1
  93. package/dist/client/index.js +871 -742
  94. package/dist/client/index.js.map +1 -1
  95. package/dist/client/providers.js +1 -1
  96. package/dist/client/styles.css +1 -1
  97. package/dist/client/utils.js +7 -7
  98. package/dist/client-bundle-stats.html +1 -1
  99. package/dist/server/index.cjs +1 -1
  100. package/dist/server/index.d.ts +5 -0
  101. package/dist/server/index.js +7 -0
  102. package/package.json +1 -1
@@ -1,6 +1,6 @@
1
1
  import { useContext as E, createContext as S, useMemo as f, useState as x, useEffect as M, useCallback as C } from "react";
2
2
  import { jsx as p } from "react/jsx-runtime";
3
- import { u as z, a as k, Q as D, b as L } from "./vendor-AVsJ2ni0.js";
3
+ import { u as z, a as k, Q as D, b as L } from "./vendor-B2EH3V58.js";
4
4
  class Q {
5
5
  apiUrl;
6
6
  headers;
@@ -551,4 +551,4 @@ export {
551
551
  Z as u,
552
552
  K as w
553
553
  };
554
- //# sourceMappingURL=providers-CgxXm6Ll.js.map
554
+ //# sourceMappingURL=providers-CCw8Kjlc.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"providers-CgxXm6Ll.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","isThumbnailCaptureAvailable","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-CCw8Kjlc.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","isThumbnailCaptureAvailable","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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"}
@@ -3,7 +3,7 @@ async function n() {
3
3
  if (!a)
4
4
  return i || (i = (async () => {
5
5
  try {
6
- const t = await import("./core-D_8mkGpQ.js"), l = await import("./javascript-DII1YQGr.js"), e = await import("./sql-IeKX8fQ8.js"), r = await import("./json-C_6Prymp.js");
6
+ const t = await import("./core-Y9e-sNfb.js"), l = await import("./javascript-DII1YQGr.js"), e = await import("./sql-IeKX8fQ8.js"), r = await import("./json-C_6Prymp.js");
7
7
  t.default.registerLanguage("javascript", l.default), t.default.registerLanguage("sql", e.default), t.default.registerLanguage("json", r.default), a = t.default;
8
8
  } catch (t) {
9
9
  console.error("Failed to load syntax highlighter:", t), i = null;
@@ -31,4 +31,4 @@ export {
31
31
  g as i,
32
32
  n as l
33
33
  };
34
- //# sourceMappingURL=syntaxHighlighting-BQfjio-i.js.map
34
+ //# sourceMappingURL=syntaxHighlighting-DAMSW_A6.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"syntaxHighlighting-BQfjio-i.js","sources":["../../../src/client/utils/syntaxHighlighting.ts"],"sourcesContent":["/**\n * Syntax Highlighting Utility\n *\n * Lazy-loads highlight.js only when needed for syntax highlighting in debug panels.\n * This minimizes bundle size impact since syntax highlighting is not needed on initial load.\n *\n * Usage:\n * await highlightCodeBlocks()\n */\n\nlet highlightJs: any = null\nlet loadingPromise: Promise<void> | null = null\n\n/**\n * Lazy-loads highlight.js and registers supported languages.\n * Only loads once - subsequent calls return immediately.\n */\nexport async function loadSyntaxHighlighter(): Promise<void> {\n // Already loaded\n if (highlightJs) {\n return\n }\n\n // Loading in progress - wait for it\n if (loadingPromise) {\n return loadingPromise\n }\n\n loadingPromise = (async () => {\n try {\n // Dynamic imports to enable code splitting\n const hljs = await import('highlight.js/lib/core')\n const javascript = await import('highlight.js/lib/languages/javascript')\n const sql = await import('highlight.js/lib/languages/sql')\n const json = await import('highlight.js/lib/languages/json')\n\n // Register languages we need\n hljs.default.registerLanguage('javascript', javascript.default)\n hljs.default.registerLanguage('sql', sql.default)\n hljs.default.registerLanguage('json', json.default)\n\n // Store the instance\n highlightJs = hljs.default\n } catch (err) {\n console.error('Failed to load syntax highlighter:', err)\n // Clear loading promise so it can be retried\n loadingPromise = null\n }\n })()\n\n return loadingPromise\n}\n\n/**\n * Highlights all code blocks on the page that haven't been highlighted yet.\n * Gracefully handles cases where highlight.js fails to load.\n */\nexport async function highlightCodeBlocks(): Promise<void> {\n // Load highlighter if not already loaded\n await loadSyntaxHighlighter()\n\n // If loading failed, return silently (code blocks remain unstyled)\n if (!highlightJs) {\n return\n }\n\n // Find all code blocks and highlight them\n document.querySelectorAll('pre code').forEach((block) => {\n // Skip if already highlighted\n if (!block.classList.contains('hljs')) {\n highlightJs.highlightElement(block)\n }\n })\n}\n\n/**\n * Highlights a specific code block element.\n * Useful for dynamically added content.\n *\n * @param element - The code block element to highlight\n */\nexport async function highlightCodeBlock(element: HTMLElement): Promise<void> {\n await loadSyntaxHighlighter()\n\n if (!highlightJs) {\n return\n }\n\n if (!element.classList.contains('hljs')) {\n highlightJs.highlightElement(element)\n }\n}\n\n/**\n * Returns whether syntax highlighting is available.\n * Useful for conditional rendering or feature detection.\n */\nexport function isSyntaxHighlightingAvailable(): boolean {\n return highlightJs !== null\n}\n\n/**\n * Returns the loaded highlight.js instance, if available.\n * Useful for custom highlighting flows that need direct access.\n */\nexport function getSyntaxHighlighter(): any | null {\n return highlightJs\n}\n"],"names":["highlightJs","loadingPromise","loadSyntaxHighlighter","hljs","javascript","sql","json","err","highlightCodeBlocks","block","highlightCodeBlock","element","isSyntaxHighlightingAvailable","getSyntaxHighlighter"],"mappings":"AAUA,IAAIA,IAAmB,MACnBC,IAAuC;AAM3C,eAAsBC,IAAuC;AAE3D,MAAI,CAAAF;AAKJ,WAAIC,MAIJA,KAAkB,YAAY;AAC5B,UAAI;AAEF,cAAME,IAAO,MAAM,OAAO,oBAAuB,GAC3CC,IAAa,MAAM,OAAO,0BAAuC,GACjEC,IAAM,MAAM,OAAO,mBAAgC,GACnDC,IAAO,MAAM,OAAO,oBAAiC;AAG3D,QAAAH,EAAK,QAAQ,iBAAiB,cAAcC,EAAW,OAAO,GAC9DD,EAAK,QAAQ,iBAAiB,OAAOE,EAAI,OAAO,GAChDF,EAAK,QAAQ,iBAAiB,QAAQG,EAAK,OAAO,GAGlDN,IAAcG,EAAK;AAAA,MACrB,SAASI,GAAK;AACZ,gBAAQ,MAAM,sCAAsCA,CAAG,GAEvDN,IAAiB;AAAA,MACnB;AAAA,IACF,GAAA,GAEOA;AACT;AAMA,eAAsBO,IAAqC;AAKzD,EAHA,MAAMN,EAAA,GAGDF,KAKL,SAAS,iBAAiB,UAAU,EAAE,QAAQ,CAACS,MAAU;AAEvD,IAAKA,EAAM,UAAU,SAAS,MAAM,KAClCT,EAAY,iBAAiBS,CAAK;AAAA,EAEtC,CAAC;AACH;AAQA,eAAsBC,EAAmBC,GAAqC;AAG5E,EAFA,MAAMT,EAAA,GAEDF,MAIAW,EAAQ,UAAU,SAAS,MAAM,KACpCX,EAAY,iBAAiBW,CAAO;AAExC;AAMO,SAASC,IAAyC;AACvD,SAAOZ,MAAgB;AACzB;AAMO,SAASa,IAAmC;AACjD,SAAOb;AACT;"}
1
+ {"version":3,"file":"syntaxHighlighting-DAMSW_A6.js","sources":["../../../src/client/utils/syntaxHighlighting.ts"],"sourcesContent":["/**\n * Syntax Highlighting Utility\n *\n * Lazy-loads highlight.js only when needed for syntax highlighting in debug panels.\n * This minimizes bundle size impact since syntax highlighting is not needed on initial load.\n *\n * Usage:\n * await highlightCodeBlocks()\n */\n\nlet highlightJs: any = null\nlet loadingPromise: Promise<void> | null = null\n\n/**\n * Lazy-loads highlight.js and registers supported languages.\n * Only loads once - subsequent calls return immediately.\n */\nexport async function loadSyntaxHighlighter(): Promise<void> {\n // Already loaded\n if (highlightJs) {\n return\n }\n\n // Loading in progress - wait for it\n if (loadingPromise) {\n return loadingPromise\n }\n\n loadingPromise = (async () => {\n try {\n // Dynamic imports to enable code splitting\n const hljs = await import('highlight.js/lib/core')\n const javascript = await import('highlight.js/lib/languages/javascript')\n const sql = await import('highlight.js/lib/languages/sql')\n const json = await import('highlight.js/lib/languages/json')\n\n // Register languages we need\n hljs.default.registerLanguage('javascript', javascript.default)\n hljs.default.registerLanguage('sql', sql.default)\n hljs.default.registerLanguage('json', json.default)\n\n // Store the instance\n highlightJs = hljs.default\n } catch (err) {\n console.error('Failed to load syntax highlighter:', err)\n // Clear loading promise so it can be retried\n loadingPromise = null\n }\n })()\n\n return loadingPromise\n}\n\n/**\n * Highlights all code blocks on the page that haven't been highlighted yet.\n * Gracefully handles cases where highlight.js fails to load.\n */\nexport async function highlightCodeBlocks(): Promise<void> {\n // Load highlighter if not already loaded\n await loadSyntaxHighlighter()\n\n // If loading failed, return silently (code blocks remain unstyled)\n if (!highlightJs) {\n return\n }\n\n // Find all code blocks and highlight them\n document.querySelectorAll('pre code').forEach((block) => {\n // Skip if already highlighted\n if (!block.classList.contains('hljs')) {\n highlightJs.highlightElement(block)\n }\n })\n}\n\n/**\n * Highlights a specific code block element.\n * Useful for dynamically added content.\n *\n * @param element - The code block element to highlight\n */\nexport async function highlightCodeBlock(element: HTMLElement): Promise<void> {\n await loadSyntaxHighlighter()\n\n if (!highlightJs) {\n return\n }\n\n if (!element.classList.contains('hljs')) {\n highlightJs.highlightElement(element)\n }\n}\n\n/**\n * Returns whether syntax highlighting is available.\n * Useful for conditional rendering or feature detection.\n */\nexport function isSyntaxHighlightingAvailable(): boolean {\n return highlightJs !== null\n}\n\n/**\n * Returns the loaded highlight.js instance, if available.\n * Useful for custom highlighting flows that need direct access.\n */\nexport function getSyntaxHighlighter(): any | null {\n return highlightJs\n}\n"],"names":["highlightJs","loadingPromise","loadSyntaxHighlighter","hljs","javascript","sql","json","err","highlightCodeBlocks","block","highlightCodeBlock","element","isSyntaxHighlightingAvailable","getSyntaxHighlighter"],"mappings":"AAUA,IAAIA,IAAmB,MACnBC,IAAuC;AAM3C,eAAsBC,IAAuC;AAE3D,MAAI,CAAAF;AAKJ,WAAIC,MAIJA,KAAkB,YAAY;AAC5B,UAAI;AAEF,cAAME,IAAO,MAAM,OAAO,oBAAuB,GAC3CC,IAAa,MAAM,OAAO,0BAAuC,GACjEC,IAAM,MAAM,OAAO,mBAAgC,GACnDC,IAAO,MAAM,OAAO,oBAAiC;AAG3D,QAAAH,EAAK,QAAQ,iBAAiB,cAAcC,EAAW,OAAO,GAC9DD,EAAK,QAAQ,iBAAiB,OAAOE,EAAI,OAAO,GAChDF,EAAK,QAAQ,iBAAiB,QAAQG,EAAK,OAAO,GAGlDN,IAAcG,EAAK;AAAA,MACrB,SAASI,GAAK;AACZ,gBAAQ,MAAM,sCAAsCA,CAAG,GAEvDN,IAAiB;AAAA,MACnB;AAAA,IACF,GAAA,GAEOA;AACT;AAMA,eAAsBO,IAAqC;AAKzD,EAHA,MAAMN,EAAA,GAGDF,KAKL,SAAS,iBAAiB,UAAU,EAAE,QAAQ,CAACS,MAAU;AAEvD,IAAKA,EAAM,UAAU,SAAS,MAAM,KAClCT,EAAY,iBAAiBS,CAAK;AAAA,EAEtC,CAAC;AACH;AAQA,eAAsBC,EAAmBC,GAAqC;AAG5E,EAFA,MAAMT,EAAA,GAEDF,MAIAW,EAAQ,UAAU,SAAS,MAAM,KACpCX,EAAY,iBAAiBW,CAAO;AAExC;AAMO,SAASC,IAAyC;AACvD,SAAOZ,MAAgB;AACzB;AAMO,SAASa,IAAmC;AACjD,SAAOb;AACT;"}
@@ -1,6 +1,6 @@
1
- import { u as W, a as U } from "./vendor-AVsJ2ni0.js";
1
+ import { u as W, a as U } from "./vendor-B2EH3V58.js";
2
2
  import { useState as A, useRef as I, useMemo as v, useEffect as q, useCallback as M } from "react";
3
- import { f as $, g as J } from "./providers-CgxXm6Ll.js";
3
+ import { f as $, g as J } from "./providers-CCw8Kjlc.js";
4
4
  import { b as ee, t as te } from "./funnel-utils-CyonoNeC.js";
5
5
  import { i as re } from "./flow-utils-CjQZG5qq.js";
6
6
  const ne = {
@@ -1203,29 +1203,29 @@ export {
1203
1203
  ne as F,
1204
1204
  We as a,
1205
1205
  Ue as b,
1206
- $e as c,
1207
- je as d,
1208
- Pe as e,
1209
- fe as f,
1210
- de as g,
1211
- Oe as h,
1206
+ X as c,
1207
+ $e as d,
1208
+ je as e,
1209
+ Pe as f,
1210
+ fe as g,
1211
+ de as h,
1212
1212
  Z as i,
1213
- Ie as j,
1214
- Ae as k,
1215
- Be as l,
1213
+ Oe as j,
1214
+ Ie as k,
1215
+ Ae as l,
1216
1216
  pe as m,
1217
- qe as n,
1218
- ze as o,
1219
- Ge as p,
1220
- xe as q,
1221
- Je as r,
1222
- Ke as s,
1223
- X as t,
1217
+ Be as n,
1218
+ qe as o,
1219
+ ze as p,
1220
+ Ge as q,
1221
+ xe as r,
1222
+ B as s,
1223
+ Je as t,
1224
1224
  ce as u,
1225
1225
  Ne as v,
1226
- B as w,
1226
+ Ke as w,
1227
1227
  Ye as x,
1228
1228
  He as y,
1229
1229
  P as z
1230
1230
  };
1231
- //# sourceMappingURL=useDirtyStateTracking-Cu1HSjmo.js.map
1231
+ //# sourceMappingURL=useDirtyStateTracking-CjhwBXRw.js.map