@forgecharts/sdk 1.1.27 → 1.1.29

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 (356) hide show
  1. package/dist/__tests__/backwardCompatibility.test.d.ts +14 -0
  2. package/dist/__tests__/backwardCompatibility.test.d.ts.map +1 -0
  3. package/dist/__tests__/candleInvariant.test.d.ts +20 -0
  4. package/dist/__tests__/candleInvariant.test.d.ts.map +1 -0
  5. package/dist/__tests__/public-api-surface.d.ts +13 -0
  6. package/dist/__tests__/public-api-surface.d.ts.map +1 -0
  7. package/dist/__tests__/timeframeBoundary.test.d.ts +17 -0
  8. package/dist/__tests__/timeframeBoundary.test.d.ts.map +1 -0
  9. package/dist/api/DrawingManager.d.ts +20 -0
  10. package/dist/api/DrawingManager.d.ts.map +1 -0
  11. package/dist/api/EventBus.d.ts +19 -0
  12. package/dist/api/EventBus.d.ts.map +1 -0
  13. package/dist/api/IndicatorDAG.d.ts +85 -0
  14. package/dist/api/IndicatorDAG.d.ts.map +1 -0
  15. package/dist/api/IndicatorRegistry.d.ts +22 -0
  16. package/dist/api/IndicatorRegistry.d.ts.map +1 -0
  17. package/dist/api/LayoutManager.d.ts +30 -0
  18. package/dist/api/LayoutManager.d.ts.map +1 -0
  19. package/dist/api/PaneManager.d.ts +42 -0
  20. package/dist/api/PaneManager.d.ts.map +1 -0
  21. package/dist/api/ReferenceAPI.d.ts +78 -0
  22. package/dist/api/ReferenceAPI.d.ts.map +1 -0
  23. package/dist/api/TChart.d.ts +345 -0
  24. package/dist/api/TChart.d.ts.map +1 -0
  25. package/{src/api/createChart.ts → dist/api/createChart.d.ts} +2 -7
  26. package/dist/api/createChart.d.ts.map +1 -0
  27. package/dist/api/drawing tools/fib gann menu/fibRetracement.d.ts +4 -0
  28. package/dist/api/drawing tools/fib gann menu/fibRetracement.d.ts.map +1 -0
  29. package/dist/api/drawing tools/lines menu/crossLine.d.ts +4 -0
  30. package/dist/api/drawing tools/lines menu/crossLine.d.ts.map +1 -0
  31. package/dist/api/drawing tools/lines menu/disjointChannel.d.ts +17 -0
  32. package/dist/api/drawing tools/lines menu/disjointChannel.d.ts.map +1 -0
  33. package/dist/api/drawing tools/lines menu/extendedLine.d.ts +4 -0
  34. package/dist/api/drawing tools/lines menu/extendedLine.d.ts.map +1 -0
  35. package/dist/api/drawing tools/lines menu/flatTopBottom.d.ts +4 -0
  36. package/dist/api/drawing tools/lines menu/flatTopBottom.d.ts.map +1 -0
  37. package/dist/api/drawing tools/lines menu/horizontal.d.ts +4 -0
  38. package/dist/api/drawing tools/lines menu/horizontal.d.ts.map +1 -0
  39. package/dist/api/drawing tools/lines menu/horizontalRay.d.ts +4 -0
  40. package/dist/api/drawing tools/lines menu/horizontalRay.d.ts.map +1 -0
  41. package/dist/api/drawing tools/lines menu/infoLine.d.ts +4 -0
  42. package/dist/api/drawing tools/lines menu/infoLine.d.ts.map +1 -0
  43. package/dist/api/drawing tools/lines menu/insidePitchfork.d.ts +4 -0
  44. package/dist/api/drawing tools/lines menu/insidePitchfork.d.ts.map +1 -0
  45. package/dist/api/drawing tools/lines menu/modifiedSchiffPitchfork.d.ts +4 -0
  46. package/dist/api/drawing tools/lines menu/modifiedSchiffPitchfork.d.ts.map +1 -0
  47. package/dist/api/drawing tools/lines menu/parallelChannel.d.ts +4 -0
  48. package/dist/api/drawing tools/lines menu/parallelChannel.d.ts.map +1 -0
  49. package/dist/api/drawing tools/lines menu/pitchfork.d.ts +4 -0
  50. package/dist/api/drawing tools/lines menu/pitchfork.d.ts.map +1 -0
  51. package/dist/api/drawing tools/lines menu/ray.d.ts +4 -0
  52. package/dist/api/drawing tools/lines menu/ray.d.ts.map +1 -0
  53. package/dist/api/drawing tools/lines menu/regressionTrend.d.ts +4 -0
  54. package/dist/api/drawing tools/lines menu/regressionTrend.d.ts.map +1 -0
  55. package/dist/api/drawing tools/lines menu/schiffPitchfork.d.ts +4 -0
  56. package/dist/api/drawing tools/lines menu/schiffPitchfork.d.ts.map +1 -0
  57. package/dist/api/drawing tools/lines menu/trendAngle.d.ts +4 -0
  58. package/dist/api/drawing tools/lines menu/trendAngle.d.ts.map +1 -0
  59. package/dist/api/drawing tools/lines menu/trendline.d.ts +4 -0
  60. package/dist/api/drawing tools/lines menu/trendline.d.ts.map +1 -0
  61. package/dist/api/drawing tools/lines menu/vertical.d.ts +4 -0
  62. package/dist/api/drawing tools/lines menu/vertical.d.ts.map +1 -0
  63. package/{src/api/drawing tools/pointers menu/crosshair.ts → dist/api/drawing tools/pointers menu/crosshair.d.ts } +8 -9
  64. package/dist/api/drawing tools/pointers menu/crosshair.d.ts.map +1 -0
  65. package/dist/api/drawing tools/pointers menu/cursor.d.ts +15 -0
  66. package/dist/api/drawing tools/pointers menu/cursor.d.ts.map +1 -0
  67. package/{src/api/drawing tools/pointers menu/demonstration.ts → dist/api/drawing tools/pointers menu/demonstration.d.ts } +12 -17
  68. package/dist/api/drawing tools/pointers menu/demonstration.d.ts.map +1 -0
  69. package/{src/api/drawing tools/pointers menu/dot.ts → dist/api/drawing tools/pointers menu/dot.d.ts } +10 -13
  70. package/dist/api/drawing tools/pointers menu/dot.d.ts.map +1 -0
  71. package/dist/api/drawing tools/shapes menu/rectangle.d.ts +4 -0
  72. package/dist/api/drawing tools/shapes menu/rectangle.d.ts.map +1 -0
  73. package/dist/api/drawing tools/shapes menu/text.d.ts +4 -0
  74. package/dist/api/drawing tools/shapes menu/text.d.ts.map +1 -0
  75. package/dist/api/drawingUtils.d.ts +22 -0
  76. package/dist/api/drawingUtils.d.ts.map +1 -0
  77. package/dist/core/CanvasLayer.d.ts +26 -0
  78. package/dist/core/CanvasLayer.d.ts.map +1 -0
  79. package/dist/core/Chart.d.ts +164 -0
  80. package/dist/core/Chart.d.ts.map +1 -0
  81. package/dist/core/CoordTransform.d.ts +168 -0
  82. package/dist/core/CoordTransform.d.ts.map +1 -0
  83. package/dist/core/Crosshair.d.ts +30 -0
  84. package/dist/core/Crosshair.d.ts.map +1 -0
  85. package/dist/core/IndicatorEngine.d.ts +51 -0
  86. package/dist/core/IndicatorEngine.d.ts.map +1 -0
  87. package/dist/core/InteractionManager.d.ts +197 -0
  88. package/dist/core/InteractionManager.d.ts.map +1 -0
  89. package/dist/core/PriceScale.d.ts +27 -0
  90. package/dist/core/PriceScale.d.ts.map +1 -0
  91. package/dist/core/Series.d.ts +40 -0
  92. package/dist/core/Series.d.ts.map +1 -0
  93. package/dist/core/TimeScale.d.ts +43 -0
  94. package/dist/core/TimeScale.d.ts.map +1 -0
  95. package/dist/datafeed/DatafeedConnector.d.ts +89 -0
  96. package/dist/datafeed/DatafeedConnector.d.ts.map +1 -0
  97. package/dist/engine/CandleEngine.d.ts +207 -0
  98. package/dist/engine/CandleEngine.d.ts.map +1 -0
  99. package/dist/engine/__tests__/CandleEngine.test.d.ts +2 -0
  100. package/dist/engine/__tests__/CandleEngine.test.d.ts.map +1 -0
  101. package/dist/engine/candleInvariants.d.ts +66 -0
  102. package/dist/engine/candleInvariants.d.ts.map +1 -0
  103. package/{src/engine/mergeUtils.ts → dist/engine/mergeUtils.d.ts} +15 -52
  104. package/dist/engine/mergeUtils.d.ts.map +1 -0
  105. package/dist/engine/timeframeUtils.d.ts +80 -0
  106. package/dist/engine/timeframeUtils.d.ts.map +1 -0
  107. package/dist/index.d.ts +40 -0
  108. package/dist/index.d.ts.map +1 -0
  109. package/dist/index.js +8101 -0
  110. package/dist/index.js.map +1 -0
  111. package/{src/internal.ts → dist/internal.d.ts} +1 -13
  112. package/dist/internal.d.ts.map +1 -0
  113. package/dist/internal.js +8852 -0
  114. package/dist/internal.js.map +1 -0
  115. package/dist/licensing/ChartRuntimeResolver.d.ts +233 -0
  116. package/dist/licensing/ChartRuntimeResolver.d.ts.map +1 -0
  117. package/dist/licensing/LicenseManager.d.ts +55 -0
  118. package/dist/licensing/LicenseManager.d.ts.map +1 -0
  119. package/dist/licensing/__tests__/ChartRuntimeResolver.test.d.ts +13 -0
  120. package/dist/licensing/__tests__/ChartRuntimeResolver.test.d.ts.map +1 -0
  121. package/dist/licensing/__tests__/LicenseManager.test.d.ts +12 -0
  122. package/dist/licensing/__tests__/LicenseManager.test.d.ts.map +1 -0
  123. package/dist/licensing/licenseTypes.d.ts +18 -0
  124. package/dist/licensing/licenseTypes.d.ts.map +1 -0
  125. package/dist/pine/PineCompiler.d.ts +35 -0
  126. package/dist/pine/PineCompiler.d.ts.map +1 -0
  127. package/dist/pine/diagnostics.d.ts +20 -0
  128. package/dist/pine/diagnostics.d.ts.map +1 -0
  129. package/{src/pine/index.ts → dist/pine/index.d.ts} +4 -3
  130. package/dist/pine/index.d.ts.map +1 -0
  131. package/dist/pine/pine-ast.d.ts +142 -0
  132. package/dist/pine/pine-ast.d.ts.map +1 -0
  133. package/dist/pine/pine-lexer.d.ts +41 -0
  134. package/dist/pine/pine-lexer.d.ts.map +1 -0
  135. package/dist/pine/pine-parser.d.ts +51 -0
  136. package/dist/pine/pine-parser.d.ts.map +1 -0
  137. package/dist/pine/pine-transpiler.d.ts +33 -0
  138. package/dist/pine/pine-transpiler.d.ts.map +1 -0
  139. package/dist/pixi/LayerName.d.ts +18 -0
  140. package/dist/pixi/LayerName.d.ts.map +1 -0
  141. package/dist/pixi/PixiCandlestickRenderer.d.ts +23 -0
  142. package/dist/pixi/PixiCandlestickRenderer.d.ts.map +1 -0
  143. package/dist/pixi/PixiChart.d.ts +72 -0
  144. package/dist/pixi/PixiChart.d.ts.map +1 -0
  145. package/dist/pixi/PixiCrosshairRenderer.d.ts +29 -0
  146. package/dist/pixi/PixiCrosshairRenderer.d.ts.map +1 -0
  147. package/dist/pixi/PixiDrawingRenderer.d.ts +17 -0
  148. package/dist/pixi/PixiDrawingRenderer.d.ts.map +1 -0
  149. package/dist/pixi/PixiGridRenderer.d.ts +22 -0
  150. package/dist/pixi/PixiGridRenderer.d.ts.map +1 -0
  151. package/dist/pixi/PixiLayerManager.d.ts +56 -0
  152. package/dist/pixi/PixiLayerManager.d.ts.map +1 -0
  153. package/dist/react/canvas/ChartCanvas.d.ts +85 -0
  154. package/dist/react/canvas/ChartCanvas.d.ts.map +1 -0
  155. package/dist/react/canvas/ChartContextMenu.d.ts +18 -0
  156. package/dist/react/canvas/ChartContextMenu.d.ts.map +1 -0
  157. package/dist/react/canvas/ChartSettingsDialog.d.ts +25 -0
  158. package/dist/react/canvas/ChartSettingsDialog.d.ts.map +1 -0
  159. package/dist/react/canvas/IndicatorLabel.d.ts +21 -0
  160. package/dist/react/canvas/IndicatorLabel.d.ts.map +1 -0
  161. package/dist/react/canvas/IndicatorPane.d.ts +32 -0
  162. package/dist/react/canvas/IndicatorPane.d.ts.map +1 -0
  163. package/dist/react/canvas/PointerOverlay.d.ts +23 -0
  164. package/dist/react/canvas/PointerOverlay.d.ts.map +1 -0
  165. package/dist/react/canvas/toolbars/LeftToolbar.d.ts +19 -0
  166. package/dist/react/canvas/toolbars/LeftToolbar.d.ts.map +1 -0
  167. package/dist/react/hooks/useChartCapabilities.d.ts +21 -0
  168. package/dist/react/hooks/useChartCapabilities.d.ts.map +1 -0
  169. package/{src/react/index.ts → dist/react/index.d.ts} +2 -23
  170. package/dist/react/index.d.ts.map +1 -0
  171. package/dist/react/index.js +11559 -0
  172. package/dist/react/index.js.map +1 -0
  173. package/{src/react/internal.ts → dist/react/internal.d.ts} +2 -26
  174. package/dist/react/internal.d.ts.map +1 -0
  175. package/dist/react/internal.js +12148 -0
  176. package/dist/react/internal.js.map +1 -0
  177. package/dist/react/shell/ManagedAppShell.d.ts +91 -0
  178. package/dist/react/shell/ManagedAppShell.d.ts.map +1 -0
  179. package/dist/react/trading/TradingBridge.d.ts +86 -0
  180. package/dist/react/trading/TradingBridge.d.ts.map +1 -0
  181. package/dist/react/workspace/ChartWorkspace.d.ts +73 -0
  182. package/dist/react/workspace/ChartWorkspace.d.ts.map +1 -0
  183. package/dist/react/workspace/FloatingPanel.d.ts +18 -0
  184. package/dist/react/workspace/FloatingPanel.d.ts.map +1 -0
  185. package/dist/react/workspace/IndicatorsDialog.d.ts +8 -0
  186. package/dist/react/workspace/IndicatorsDialog.d.ts.map +1 -0
  187. package/dist/react/workspace/LayoutMenu.d.ts +33 -0
  188. package/dist/react/workspace/LayoutMenu.d.ts.map +1 -0
  189. package/dist/react/workspace/SymbolSearchDialog.d.ts +10 -0
  190. package/dist/react/workspace/SymbolSearchDialog.d.ts.map +1 -0
  191. package/dist/react/workspace/TabBar.d.ts +17 -0
  192. package/dist/react/workspace/TabBar.d.ts.map +1 -0
  193. package/dist/react/workspace/toolbars/BottomToolbar.d.ts +19 -0
  194. package/dist/react/workspace/toolbars/BottomToolbar.d.ts.map +1 -0
  195. package/dist/react/workspace/toolbars/RightToolbar.d.ts +8 -0
  196. package/dist/react/workspace/toolbars/RightToolbar.d.ts.map +1 -0
  197. package/dist/react/workspace/toolbars/TopToolbar.d.ts +41 -0
  198. package/dist/react/workspace/toolbars/TopToolbar.d.ts.map +1 -0
  199. package/dist/renderers/CandlestickRenderer.d.ts +13 -0
  200. package/dist/renderers/CandlestickRenderer.d.ts.map +1 -0
  201. package/dist/renderers/HistogramRenderer.d.ts +11 -0
  202. package/dist/renderers/HistogramRenderer.d.ts.map +1 -0
  203. package/dist/renderers/LineRenderer.d.ts +12 -0
  204. package/dist/renderers/LineRenderer.d.ts.map +1 -0
  205. package/dist/theme/colors.d.ts +4 -0
  206. package/dist/theme/colors.d.ts.map +1 -0
  207. package/dist/tools/barDivergenceCheck.d.ts +120 -0
  208. package/dist/tools/barDivergenceCheck.d.ts.map +1 -0
  209. package/dist/trading/TradingOverlayStore.d.ts +86 -0
  210. package/dist/trading/TradingOverlayStore.d.ts.map +1 -0
  211. package/dist/trading/UnmanagedIngestion.d.ts +91 -0
  212. package/dist/trading/UnmanagedIngestion.d.ts.map +1 -0
  213. package/dist/trading/__tests__/ManagedTradingController.test.d.ts +18 -0
  214. package/dist/trading/__tests__/ManagedTradingController.test.d.ts.map +1 -0
  215. package/dist/trading/__tests__/TradingOverlayStore.test.d.ts +16 -0
  216. package/dist/trading/__tests__/TradingOverlayStore.test.d.ts.map +1 -0
  217. package/dist/trading/__tests__/UnmanagedIngestion.test.d.ts +16 -0
  218. package/dist/trading/__tests__/UnmanagedIngestion.test.d.ts.map +1 -0
  219. package/dist/trading/managed/ManagedTradingController.d.ts +110 -0
  220. package/dist/trading/managed/ManagedTradingController.d.ts.map +1 -0
  221. package/dist/trading/managed/managedCapabilities.d.ts +45 -0
  222. package/dist/trading/managed/managedCapabilities.d.ts.map +1 -0
  223. package/dist/trading/managed/managedTypes.d.ts +122 -0
  224. package/dist/trading/managed/managedTypes.d.ts.map +1 -0
  225. package/dist/trading/tradingTypes.d.ts +89 -0
  226. package/dist/trading/tradingTypes.d.ts.map +1 -0
  227. package/dist/tscript/TScriptIndicator.d.ts +41 -0
  228. package/dist/tscript/TScriptIndicator.d.ts.map +1 -0
  229. package/dist/tscript/ast.d.ts +89 -0
  230. package/dist/tscript/ast.d.ts.map +1 -0
  231. package/dist/tscript/lexer.d.ts +36 -0
  232. package/dist/tscript/lexer.d.ts.map +1 -0
  233. package/dist/tscript/parser.d.ts +50 -0
  234. package/dist/tscript/parser.d.ts.map +1 -0
  235. package/dist/tscript/runtime.d.ts +123 -0
  236. package/dist/tscript/runtime.d.ts.map +1 -0
  237. package/dist/tscript/series.d.ts +49 -0
  238. package/dist/tscript/series.d.ts.map +1 -0
  239. package/dist/types/IChart.d.ts +48 -0
  240. package/dist/types/IChart.d.ts.map +1 -0
  241. package/{src/types/IRenderer.ts → dist/types/IRenderer.d.ts} +2 -8
  242. package/dist/types/IRenderer.d.ts.map +1 -0
  243. package/dist/types/ISeries.d.ts +26 -0
  244. package/dist/types/ISeries.d.ts.map +1 -0
  245. package/package.json +5 -1
  246. package/src/__tests__/backwardCompatibility.test.ts +0 -191
  247. package/src/__tests__/candleInvariant.test.ts +0 -500
  248. package/src/__tests__/public-api-surface.ts +0 -76
  249. package/src/__tests__/timeframeBoundary.test.ts +0 -583
  250. package/src/api/DrawingManager.ts +0 -188
  251. package/src/api/EventBus.ts +0 -53
  252. package/src/api/IndicatorDAG.ts +0 -389
  253. package/src/api/IndicatorRegistry.ts +0 -47
  254. package/src/api/LayoutManager.ts +0 -72
  255. package/src/api/PaneManager.ts +0 -129
  256. package/src/api/ReferenceAPI.ts +0 -195
  257. package/src/api/TChart.ts +0 -881
  258. package/src/api/drawing tools/fib gann menu/fibRetracement.ts +0 -27
  259. package/src/api/drawing tools/lines menu/crossLine.ts +0 -21
  260. package/src/api/drawing tools/lines menu/disjointChannel.ts +0 -74
  261. package/src/api/drawing tools/lines menu/extendedLine.ts +0 -22
  262. package/src/api/drawing tools/lines menu/flatTopBottom.ts +0 -45
  263. package/src/api/drawing tools/lines menu/horizontal.ts +0 -24
  264. package/src/api/drawing tools/lines menu/horizontalRay.ts +0 -25
  265. package/src/api/drawing tools/lines menu/infoLine.ts +0 -127
  266. package/src/api/drawing tools/lines menu/insidePitchfork.ts +0 -21
  267. package/src/api/drawing tools/lines menu/modifiedSchiffPitchfork.ts +0 -18
  268. package/src/api/drawing tools/lines menu/parallelChannel.ts +0 -47
  269. package/src/api/drawing tools/lines menu/pitchfork.ts +0 -15
  270. package/src/api/drawing tools/lines menu/ray.ts +0 -28
  271. package/src/api/drawing tools/lines menu/regressionTrend.ts +0 -157
  272. package/src/api/drawing tools/lines menu/schiffPitchfork.ts +0 -18
  273. package/src/api/drawing tools/lines menu/trendAngle.ts +0 -64
  274. package/src/api/drawing tools/lines menu/trendline.ts +0 -16
  275. package/src/api/drawing tools/lines menu/vertical.ts +0 -16
  276. package/src/api/drawing tools/pointers menu/cursor.ts +0 -16
  277. package/src/api/drawing tools/shapes menu/rectangle.ts +0 -24
  278. package/src/api/drawing tools/shapes menu/text.ts +0 -30
  279. package/src/api/drawingUtils.ts +0 -82
  280. package/src/core/CanvasLayer.ts +0 -77
  281. package/src/core/Chart.ts +0 -917
  282. package/src/core/CoordTransform.ts +0 -282
  283. package/src/core/Crosshair.ts +0 -207
  284. package/src/core/IndicatorEngine.ts +0 -216
  285. package/src/core/InteractionManager.ts +0 -899
  286. package/src/core/PriceScale.ts +0 -133
  287. package/src/core/Series.ts +0 -132
  288. package/src/core/TimeScale.ts +0 -175
  289. package/src/datafeed/DatafeedConnector.ts +0 -300
  290. package/src/engine/CandleEngine.ts +0 -458
  291. package/src/engine/__tests__/CandleEngine.test.ts +0 -402
  292. package/src/engine/candleInvariants.ts +0 -172
  293. package/src/engine/timeframeUtils.ts +0 -118
  294. package/src/index.ts +0 -190
  295. package/src/licensing/ChartRuntimeResolver.ts +0 -380
  296. package/src/licensing/LicenseManager.ts +0 -131
  297. package/src/licensing/__tests__/ChartRuntimeResolver.test.ts +0 -207
  298. package/src/licensing/__tests__/LicenseManager.test.ts +0 -180
  299. package/src/licensing/licenseTypes.ts +0 -19
  300. package/src/pine/PineCompiler.ts +0 -68
  301. package/src/pine/diagnostics.ts +0 -30
  302. package/src/pine/pine-ast.ts +0 -163
  303. package/src/pine/pine-lexer.ts +0 -265
  304. package/src/pine/pine-parser.ts +0 -439
  305. package/src/pine/pine-transpiler.ts +0 -301
  306. package/src/pixi/LayerName.ts +0 -35
  307. package/src/pixi/PixiCandlestickRenderer.ts +0 -125
  308. package/src/pixi/PixiChart.ts +0 -425
  309. package/src/pixi/PixiCrosshairRenderer.ts +0 -134
  310. package/src/pixi/PixiDrawingRenderer.ts +0 -121
  311. package/src/pixi/PixiGridRenderer.ts +0 -136
  312. package/src/pixi/PixiLayerManager.ts +0 -102
  313. package/src/react/canvas/ChartCanvas.tsx +0 -984
  314. package/src/react/canvas/ChartContextMenu.tsx +0 -60
  315. package/src/react/canvas/ChartSettingsDialog.tsx +0 -133
  316. package/src/react/canvas/IndicatorLabel.tsx +0 -347
  317. package/src/react/canvas/IndicatorPane.tsx +0 -503
  318. package/src/react/canvas/PointerOverlay.tsx +0 -126
  319. package/src/react/canvas/toolbars/LeftToolbar.tsx +0 -1096
  320. package/src/react/hooks/useChartCapabilities.ts +0 -76
  321. package/src/react/shell/ManagedAppShell.tsx +0 -699
  322. package/src/react/trading/TradingBridge.ts +0 -156
  323. package/src/react/workspace/ChartWorkspace.tsx +0 -228
  324. package/src/react/workspace/FloatingPanel.tsx +0 -131
  325. package/src/react/workspace/IndicatorsDialog.tsx +0 -246
  326. package/src/react/workspace/LayoutMenu.tsx +0 -345
  327. package/src/react/workspace/SymbolSearchDialog.tsx +0 -377
  328. package/src/react/workspace/TabBar.tsx +0 -87
  329. package/src/react/workspace/toolbars/BottomToolbar.tsx +0 -372
  330. package/src/react/workspace/toolbars/RightToolbar.tsx +0 -46
  331. package/src/react/workspace/toolbars/TopToolbar.tsx +0 -431
  332. package/src/renderers/CandlestickRenderer.ts +0 -130
  333. package/src/renderers/HistogramRenderer.ts +0 -63
  334. package/src/renderers/LineRenderer.ts +0 -77
  335. package/src/theme/colors.ts +0 -21
  336. package/src/tools/barDivergenceCheck.ts +0 -305
  337. package/src/trading/TradingOverlayStore.ts +0 -161
  338. package/src/trading/UnmanagedIngestion.ts +0 -156
  339. package/src/trading/__tests__/ManagedTradingController.test.ts +0 -338
  340. package/src/trading/__tests__/TradingOverlayStore.test.ts +0 -323
  341. package/src/trading/__tests__/UnmanagedIngestion.test.ts +0 -205
  342. package/src/trading/managed/ManagedTradingController.ts +0 -292
  343. package/src/trading/managed/managedCapabilities.ts +0 -98
  344. package/src/trading/managed/managedTypes.ts +0 -151
  345. package/src/trading/tradingTypes.ts +0 -135
  346. package/src/tscript/TScriptIndicator.ts +0 -54
  347. package/src/tscript/ast.ts +0 -105
  348. package/src/tscript/lexer.ts +0 -190
  349. package/src/tscript/parser.ts +0 -334
  350. package/src/tscript/runtime.ts +0 -525
  351. package/src/tscript/series.ts +0 -84
  352. package/src/types/IChart.ts +0 -56
  353. package/src/types/ISeries.ts +0 -30
  354. package/tsconfig.json +0 -23
  355. package/tsup.config.ts +0 -16
  356. package/vitest.config.ts +0 -25
@@ -1,699 +0,0 @@
1
- /**
2
- * ManagedAppShell — stateful chart application shell.
3
- *
4
- * Owns all workspace state (tabs, symbol, timeframe, layout persistence, etc.)
5
- * and wires injected API clients to ChartWorkspace. App-specific drawers (script,
6
- * watchlist, trade) are passed as render-prop factories so this shell never
7
- * imports from the app layer.
8
- */
9
-
10
- import React, {
11
- useState,
12
- useEffect,
13
- useRef,
14
- useCallback,
15
- forwardRef,
16
- useImperativeHandle,
17
- } from 'react';
18
- import { ChartCanvas } from '../canvas/ChartCanvas';
19
- import type { ChartCanvasHandle } from '../canvas/ChartCanvas';
20
- import { ChartWorkspace } from '../workspace/ChartWorkspace';
21
- import { DEFAULT_FAVORITES } from '../workspace/toolbars/TopToolbar';
22
- import type { LayoutRecord } from '../workspace/LayoutMenu';
23
- import { useChartCapabilities } from '../hooks/useChartCapabilities';
24
- import { LicenseManager } from '../../';
25
- import type { LicensePayload } from '../../';
26
- import type { IDatafeed, ChartTheme, ChartLayout, IndicatorConfig, ISymbolResolver, ChartRuntimeConfig } from '@forgecharts/types';
27
- import type { TradingBridgeCallbacks } from '../trading/TradingBridge';
28
- import type { TradingSession } from '../workspace/toolbars/BottomToolbar';
29
-
30
- // ─── API surface injected from the host app ───────────────────────────────────
31
-
32
- export type LayoutsAPI = {
33
- list: () => Promise<LayoutRecord[]>;
34
- get: (id: string) => Promise<LayoutRecord>;
35
- create: (data: { name: string; symbol: string; timeframe: string; config: Record<string, unknown> }) => Promise<LayoutRecord>;
36
- update: (id: string, data: Partial<{ name: string; symbol: string; timeframe: string; config: Record<string, unknown> }>) => Promise<LayoutRecord>;
37
- remove: (id: string) => Promise<void>;
38
- };
39
-
40
- export type PreferencesAPI = {
41
- get: <T = Record<string, unknown>>() => Promise<T>;
42
- put: (data: Record<string, unknown>) => Promise<void>;
43
- };
44
-
45
- // ─── Component props ──────────────────────────────────────────────────────────
46
-
47
- export type ManagedAppShellProps = {
48
- datafeed: IDatafeed;
49
- layouts: LayoutsAPI;
50
- preferences: PreferencesAPI;
51
- /** Resolver used by the built-in symbol search dialog. */
52
- symbolResolver: ISymbolResolver;
53
- /**
54
- * Async callback that returns the current user's Bearer token.
55
- * Used to authenticate the internal `/api/license` fetch and can be
56
- * forwarded to `TChart` / `ReferenceAPI` instances that need auth.
57
- * If omitted, requests are sent without `Authorization` headers.
58
- */
59
- getAuthToken?: () => string | Promise<string>;
60
- /**
61
- * Base URL of the ForgeCharts API, e.g. `"https://charts.myapp.com"`.
62
- * Defaults to the current origin. Set this when the chart is embedded
63
- * on a different domain from the API server.
64
- */
65
- apiUrl?: string;
66
- /**
67
- * Optional callback to resolve an exchange logo image URL.
68
- * Receives the exchange code (e.g. "BINANCE") and returns a URL string or
69
- * null/undefined to suppress the logo. If omitted, no logo is shown.
70
- * ForgeCharts Studio sets this to the /api/logos/ endpoint.
71
- */
72
- getExchangeLogoUrl?: (exchange: string) => string | null | undefined;
73
- /**
74
- * Optional host-level feature overrides applied on top of the license.
75
- * Set a flag to `false` to suppress a feature even when the license permits
76
- * it. Omitting the prop (or individual fields) defers fully to the license.
77
- */
78
- config?: ChartRuntimeConfig;
79
- /**
80
- * Optional trading bridge callbacks threaded to every ChartCanvas instance.
81
- *
82
- * Provide this when `enableOrderEntry` / `enableDraggableOrders` etc. are
83
- * active so the chart can emit intents to your broker layer.
84
- * Use `createTradingBridgeLogger(bridge)` in dev for structured logging.
85
- */
86
- tradingBridge?: TradingBridgeCallbacks;
87
- renderScriptDrawer?: (props: { onClose: () => void; onAddIndicator: (c: IndicatorConfig) => void }) => React.ReactElement | null;
88
- renderWatchlistDrawer?: (props: { onClose: () => void; onSelectSymbol: (sym: string) => void }) => React.ReactElement | null;
89
- renderTradeDrawer?: (props: { onClose: () => void }) => React.ReactElement | null;
90
- };
91
-
92
- // ─── Imperative handle (optional — exposes active chart handle) ───────────────
93
-
94
- export type ManagedAppShellHandle = {
95
- getActiveChartHandle: () => ChartCanvasHandle | null;
96
- };
97
-
98
- // ─── Helpers ──────────────────────────────────────────────────────────────────
99
-
100
- function uid() { return Math.random().toString(36).slice(2, 10); }
101
-
102
- const DEFAULT_INDICATORS: IndicatorConfig[] = [];
103
-
104
- // ─── License loaders ──────────────────────────────────────────────────────────
105
-
106
- /**
107
- * Fetches the license payload from the API and loads it into LicenseManager.
108
- *
109
- * @param apiUrl Optional base URL (defaults to current origin).
110
- * @param getAuthToken Optional async callback to obtain a Bearer token.
111
- */
112
- async function loadLicenseFromApi(
113
- apiUrl?: string,
114
- getAuthToken?: () => string | Promise<string>,
115
- ): Promise<void> {
116
- try {
117
- const url = `${apiUrl ?? ''}/api/license`;
118
- const headers: Record<string, string> = {};
119
- if (getAuthToken) {
120
- try {
121
- const tok = await getAuthToken();
122
- if (tok) headers['Authorization'] = `Bearer ${tok}`;
123
- } catch { /* token fetch failed — continue unauthenticated */ }
124
- }
125
- const r = await fetch(url, { headers });
126
- const data = r.ok
127
- ? (await r.json() as { licensePayload: unknown })
128
- : { licensePayload: null };
129
- if (data.licensePayload && typeof data.licensePayload === 'object') {
130
- LicenseManager.getInstance().loadLicense(data.licensePayload as LicensePayload);
131
- }
132
- // If the endpoint returns no payload, leave LicenseManager in its current
133
- // state (unmanaged mode) rather than clearing a previously loaded license.
134
- } catch {
135
- // Network failure — leave LicenseManager in its current state.
136
- }
137
- }
138
-
139
- // ─── Tab type ─────────────────────────────────────────────────────────────────
140
-
141
- type Tab = {
142
- id: string;
143
- label: string;
144
- symbol: string;
145
- timeframe: string;
146
- snapshot?: ChartLayout;
147
- savedLayoutId?: string;
148
- remountCount: number;
149
- };
150
-
151
- function makeTab(n: number): Tab {
152
- return { id: uid(), label: `Chart ${n}`, symbol: 'BINANCE:BTCUSDT', timeframe: '1h', remountCount: 0 };
153
- }
154
-
155
- // ─── Workspace persistence shape ─────────────────────────────────────────────
156
-
157
- type PersistedTab = Omit<Tab, 'remountCount'>;
158
-
159
- type WorkspacePrefs = {
160
- tabs: PersistedTab[];
161
- activeTabId: string;
162
- autoSave: boolean;
163
- timezone: string;
164
- customTimeframes: string[];
165
- favoriteTfs: string[];
166
- session: TradingSession;
167
- theme: ChartTheme;
168
- };
169
-
170
- const DEFAULT_TAB = makeTab(1) satisfies Tab;
171
- const DEFAULT_WORKSPACE: WorkspacePrefs = {
172
- tabs: [{ id: DEFAULT_TAB.id, label: DEFAULT_TAB.label, symbol: DEFAULT_TAB.symbol, timeframe: DEFAULT_TAB.timeframe }],
173
- activeTabId: DEFAULT_TAB.id,
174
- autoSave: false,
175
- timezone: 'UTC',
176
- customTimeframes: [],
177
- favoriteTfs: DEFAULT_FAVORITES,
178
- session: 'regular',
179
- theme: 'dark',
180
- };
181
-
182
- function workspaceFromPrefs(raw: unknown): WorkspacePrefs {
183
- if (!raw || typeof raw !== 'object') return DEFAULT_WORKSPACE;
184
- const p = raw as Record<string, unknown>;
185
- const tabs = Array.isArray(p['tabs']) && p['tabs'].length > 0
186
- ? (p['tabs'] as PersistedTab[])
187
- : DEFAULT_WORKSPACE.tabs;
188
- const activeTabId = typeof p['activeTabId'] === 'string' && tabs.some((t) => t.id === p['activeTabId'])
189
- ? p['activeTabId']
190
- : tabs[0]!.id;
191
- return {
192
- tabs,
193
- activeTabId,
194
- autoSave: typeof p['autoSave'] === 'boolean' ? p['autoSave'] : false,
195
- timezone: typeof p['timezone'] === 'string' ? p['timezone'] : 'UTC',
196
- customTimeframes: Array.isArray(p['customTimeframes']) ? p['customTimeframes'] : [],
197
- favoriteTfs: Array.isArray(p['favoriteTfs']) && p['favoriteTfs'].length > 0
198
- ? (p['favoriteTfs'] as string[])
199
- : DEFAULT_FAVORITES,
200
- session: p['session'] === 'extended' ? 'extended' : 'regular',
201
- theme: p['theme'] === 'light' ? 'light' : 'dark',
202
- };
203
- }
204
-
205
- // ─── ManagedAppShell ──────────────────────────────────────────────────────────
206
-
207
- export const ManagedAppShell = forwardRef<ManagedAppShellHandle, ManagedAppShellProps>(
208
- function ManagedAppShell(
209
- { datafeed, layouts, preferences, symbolResolver, getExchangeLogoUrl, config,
210
- getAuthToken, apiUrl, tradingBridge, renderScriptDrawer, renderWatchlistDrawer, renderTradeDrawer },
211
- ref,
212
- ) {
213
- const containerRef = useRef<HTMLDivElement>(null);
214
-
215
- const [theme, setTheme] = useState<ChartTheme>('dark');
216
- const [wsLoaded, setWsLoaded] = useState(false);
217
- const [timezone, setTimezone] = useState(DEFAULT_WORKSPACE.timezone);
218
- const [autoSave, setAutoSave] = useState(DEFAULT_WORKSPACE.autoSave);
219
- const [tabs, setTabs] = useState<Tab[]>(() =>
220
- DEFAULT_WORKSPACE.tabs.map((t) => ({ ...t, remountCount: 0 })),
221
- );
222
- const [activeTabId, setActiveTabId] = useState(DEFAULT_WORKSPACE.activeTabId);
223
- const [customTimeframes, setCustomTimeframes] = useState<string[]>(DEFAULT_WORKSPACE.customTimeframes);
224
- const [favoriteTfs, setFavoriteTfs] = useState<string[]>(DEFAULT_WORKSPACE.favoriteTfs);
225
- const [session, setSession] = useState<TradingSession>(DEFAULT_WORKSPACE.session);
226
- const [scriptDrawerOpen, setScriptDrawerOpen] = useState(false);
227
- const [watchlistOpen, setWatchlistOpen] = useState(false);
228
- const [tradeDrawerOpen, setTradeDrawerOpen] = useState(false);
229
- const [chartDirty, setChartDirty] = useState(0);
230
- const [isFullscreen, setIsFullscreen] = useState(false);
231
-
232
- const capabilities = useChartCapabilities(config);
233
-
234
- // ── Init warnings (dev only) ───────────────────────────────────────────────
235
- useEffect(() => {
236
- if (process.env.NODE_ENV === 'production') return;
237
- if (
238
- (capabilities.orderEntry || capabilities.draggableOrders || capabilities.renderManagedTradingControls) &&
239
- !renderTradeDrawer
240
- ) {
241
- console.warn(
242
- '[ManagedAppShell] Trading capabilities are active ' +
243
- '(orderEntry=%s, draggableOrders=%s, managedTrading=%s) but ' +
244
- 'renderTradeDrawer was not provided. Trading UI will be suppressed.',
245
- capabilities.orderEntry,
246
- capabilities.draggableOrders,
247
- capabilities.managedTrading,
248
- );
249
- }
250
- // Warn once on mount and whenever trading caps or the slot presence changes.
251
- // eslint-disable-next-line react-hooks/exhaustive-deps
252
- }, [capabilities.orderEntry, capabilities.draggableOrders, capabilities.managedTrading]);
253
-
254
- const [isLicensed, setIsLicensed] = useState(() => LicenseManager.getInstance().getLicense() !== null);
255
- useEffect(() => LicenseManager.getInstance().subscribe(() => {
256
- setIsLicensed(LicenseManager.getInstance().getLicense() !== null);
257
- }), []);
258
-
259
- // Per-tab chart instance registry — all tabs remain mounted simultaneously
260
- const chartInstancesRef = useRef<Map<string, ChartCanvasHandle>>(new Map());
261
- const chartHandleRef = useRef<ChartCanvasHandle | null>(null);
262
-
263
- useImperativeHandle(ref, () => ({
264
- getActiveChartHandle: () => chartHandleRef.current,
265
- }));
266
-
267
- useEffect(() => { document.documentElement.dataset.theme = theme; }, [theme]);
268
-
269
- // ── Load workspace from DB on startup ──────────────────────────────────────
270
- useEffect(() => {
271
- preferences.get<Record<string, unknown>>().then((data) => {
272
- const ws = workspaceFromPrefs(data);
273
- setTabs(ws.tabs.map((t) => ({ ...t, remountCount: 0 })));
274
- setActiveTabId(ws.activeTabId);
275
- setAutoSave(ws.autoSave);
276
- setTimezone(ws.timezone);
277
- setCustomTimeframes(ws.customTimeframes);
278
- setFavoriteTfs(ws.favoriteTfs);
279
- setSession(ws.session);
280
- setTheme(ws.theme);
281
- if (data['licensePayload'] && typeof data['licensePayload'] === 'object') {
282
- LicenseManager.getInstance().loadLicense(data['licensePayload'] as LicensePayload);
283
- } else {
284
- void loadLicenseFromApi(apiUrl, getAuthToken);
285
- }
286
- }).catch(() => {
287
- void loadLicenseFromApi(apiUrl, getAuthToken);
288
- }).finally(() => {
289
- setWsLoaded(true);
290
- });
291
- // eslint-disable-next-line react-hooks/exhaustive-deps
292
- }, []); // run once on mount
293
-
294
- // ── Debounced workspace save to DB ─────────────────────────────────────────
295
- const saveTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
296
-
297
- // Stable refs so the debounced callback always sees latest values
298
- const activeTabIdRef = useRef(activeTabId);
299
- const autoSaveRef = useRef(autoSave);
300
- const timezoneRef = useRef(timezone);
301
- const customTimeframesRef = useRef(customTimeframes);
302
- const favoriteTfsRef = useRef(favoriteTfs);
303
- const sessionRef = useRef(session);
304
- const themeRef = useRef(theme);
305
- activeTabIdRef.current = activeTabId;
306
- autoSaveRef.current = autoSave;
307
- timezoneRef.current = timezone;
308
- customTimeframesRef.current = customTimeframes;
309
- favoriteTfsRef.current = favoriteTfs;
310
- sessionRef.current = session;
311
- themeRef.current = theme;
312
-
313
- const saveWorkspace = useCallback(() => {
314
- if (!wsLoaded) return;
315
- if (saveTimerRef.current) clearTimeout(saveTimerRef.current);
316
- saveTimerRef.current = setTimeout(() => {
317
- const snap = chartHandleRef.current?.getLayoutSnapshot();
318
- setTabs((prev) => {
319
- const persisted: PersistedTab[] = prev.map(({ remountCount: _r, ...rest }) =>
320
- rest.id === activeTabIdRef.current && snap ? { ...rest, snapshot: snap } : rest,
321
- );
322
- const ws: WorkspacePrefs = {
323
- tabs: persisted,
324
- activeTabId: activeTabIdRef.current,
325
- autoSave: autoSaveRef.current,
326
- timezone: timezoneRef.current,
327
- customTimeframes: customTimeframesRef.current,
328
- favoriteTfs: favoriteTfsRef.current,
329
- session: sessionRef.current,
330
- theme: themeRef.current,
331
- };
332
- const license = LicenseManager.getInstance().getLicense();
333
- preferences.put({ ...ws, ...(license ? { licensePayload: license } : {}) } as unknown as Record<string, unknown>)
334
- .catch(() => { /* silent — non-fatal */ });
335
- return prev;
336
- });
337
- }, 1_500);
338
- }, [wsLoaded, preferences]);
339
-
340
- // Keep chartHandleRef in sync with active tab
341
- useEffect(() => {
342
- chartHandleRef.current = chartInstancesRef.current.get(activeTabId) ?? null;
343
- }, [activeTabId]);
344
-
345
- // Trigger save whenever workspace state changes
346
- useEffect(() => { saveWorkspace(); }, [tabs, activeTabId, autoSave, timezone, customTimeframes, favoriteTfs, session, theme, chartDirty, saveWorkspace]);
347
-
348
- // Track browser fullscreen changes
349
- useEffect(() => {
350
- const handler = () => setIsFullscreen(!!document.fullscreenElement);
351
- document.addEventListener('fullscreenchange', handler);
352
- return () => document.removeEventListener('fullscreenchange', handler);
353
- }, []);
354
-
355
- const handleFullscreen = useCallback(() => {
356
- if (!document.fullscreenElement) {
357
- containerRef.current?.requestFullscreen().catch(() => {});
358
- } else {
359
- document.exitFullscreen().catch(() => {});
360
- }
361
- }, []);
362
-
363
- const handleCopyScreenshot = useCallback(async () => {
364
- const url = await chartHandleRef.current?.captureScreenshot();
365
- if (!url) return;
366
- try {
367
- const resp = await fetch(url);
368
- const blob = await resp.blob();
369
- await navigator.clipboard.write([new ClipboardItem({ 'image/png': blob })]);
370
- } catch { /* Clipboard API unavailable */ }
371
- }, []);
372
-
373
- const handleDownloadScreenshot = useCallback(async () => {
374
- const url = await chartHandleRef.current?.captureScreenshot();
375
- if (!url) return;
376
- const a = document.createElement('a');
377
- a.href = url;
378
- a.download = `forgecharts-${Date.now()}.png`;
379
- a.click();
380
- }, []);
381
-
382
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
383
- const activeTab = (tabs.find((t) => t.id === activeTabId) ?? tabs[0])!;
384
-
385
- const captureSnapshot = useCallback((): ChartLayout | null =>
386
- chartHandleRef.current?.getLayoutSnapshot() ?? null,
387
- []);
388
-
389
- // ── Tab management ─────────────────────────────────────────────────────────
390
-
391
- const selectTab = useCallback((id: string) => {
392
- const snap = captureSnapshot();
393
- setTabs((prev) =>
394
- prev.map((t): Tab => t.id === activeTabId ? { ...t, ...(snap !== null ? { snapshot: snap } : {}) } : t),
395
- );
396
- setActiveTabId(id);
397
- }, [activeTabId, captureSnapshot]);
398
-
399
- const addTab = useCallback(() => {
400
- const snap = captureSnapshot();
401
- setTabs((prev) => {
402
- const updated: Tab[] = prev.map((t): Tab =>
403
- t.id === activeTabId ? { ...t, ...(snap !== null ? { snapshot: snap } : {}) } : t,
404
- );
405
- const newTab = makeTab(updated.length + 1);
406
- setActiveTabId(newTab.id);
407
- return [...updated, newTab];
408
- });
409
- }, [activeTabId, captureSnapshot]);
410
-
411
- const closeTab = useCallback((id: string) => {
412
- setTabs((prev) => {
413
- if (prev.length <= 1) return prev;
414
- const remaining = prev.filter((t) => t.id !== id);
415
- if (id === activeTabId) {
416
- const closedIdx = prev.findIndex((t) => t.id === id);
417
- const next = remaining[Math.min(closedIdx, remaining.length - 1)];
418
- if (next) setActiveTabId(next.id);
419
- }
420
- return remaining;
421
- });
422
- }, [activeTabId]);
423
-
424
- const renameTab = useCallback((id: string, label: string) => {
425
- setTabs((prev) => prev.map((t) => (t.id === id ? { ...t, label } : t)));
426
- }, []);
427
-
428
- // ── Symbol / timeframe ─────────────────────────────────────────────────────
429
-
430
- const handleSymbolChange = useCallback((symbol: string) => {
431
- setTabs((prev) => prev.map((t) => (t.id === activeTabId ? { ...t, symbol } : t)));
432
- }, [activeTabId]);
433
-
434
- const handleTimeframeChange = useCallback((timeframe: string) => {
435
- setTabs((prev) => prev.map((t) => (t.id === activeTabId ? { ...t, timeframe } : t)));
436
- }, [activeTabId]);
437
-
438
- const handleAddCustomTimeframe = useCallback((tf: string) => {
439
- setCustomTimeframes((prev) => prev.includes(tf) ? prev : [...prev, tf]);
440
- }, []);
441
-
442
- const handleAddIndicator = useCallback((config: IndicatorConfig) => {
443
- chartHandleRef.current?.addIndicator(config);
444
- }, []);
445
-
446
- const toggleTheme = useCallback(() => setTheme((t) => (t === 'dark' ? 'light' : 'dark')), []);
447
-
448
- // ── Layout helpers ─────────────────────────────────────────────────────────
449
-
450
- const buildConfig = useCallback(
451
- (snap: ChartLayout): Record<string, unknown> => ({
452
- chartLayout: snap as unknown as Record<string, unknown>,
453
- customTimeframes,
454
- timezone,
455
- }),
456
- [customTimeframes, timezone],
457
- );
458
-
459
- const applyConfig = useCallback((cfg: Record<string, unknown>) => {
460
- const chartLayout = (
461
- cfg['chartLayout'] != null ? cfg['chartLayout'] : cfg
462
- ) as unknown as ChartLayout;
463
- if (Array.isArray(cfg['customTimeframes'])) {
464
- setCustomTimeframes(cfg['customTimeframes'] as string[]);
465
- }
466
- if (typeof cfg['timezone'] === 'string') {
467
- setTimezone(cfg['timezone']);
468
- }
469
- return chartLayout;
470
- }, []);
471
-
472
- // ── Layout save ────────────────────────────────────────────────────────────
473
-
474
- const handleSaveLayout = useCallback(async (name: string) => {
475
- const snap = captureSnapshot();
476
- if (!snap) throw new Error('No chart snapshot available');
477
- const config = buildConfig(snap);
478
- if (activeTab.savedLayoutId) {
479
- await layouts.update(activeTab.savedLayoutId, {
480
- name, symbol: activeTab.symbol, timeframe: activeTab.timeframe, config,
481
- });
482
- setTabs((prev) => prev.map((t) => (t.id === activeTabId ? { ...t, label: name } : t)));
483
- } else {
484
- const record = await layouts.create({ name, symbol: activeTab.symbol, timeframe: activeTab.timeframe, config });
485
- setTabs((prev) => prev.map((t) =>
486
- t.id === activeTabId ? { ...t, savedLayoutId: record.id, label: name } : t,
487
- ));
488
- }
489
- }, [activeTab, activeTabId, buildConfig, captureSnapshot, layouts]);
490
-
491
- // ── Layout load ────────────────────────────────────────────────────────────
492
-
493
- const handleLoadLayout = useCallback(async (layoutId: string) => {
494
- const record = await layouts.get(layoutId);
495
- const chartLayout = applyConfig(record.config);
496
- setTabs((prev) => prev.map((t) =>
497
- t.id === activeTabId
498
- ? { ...t, symbol: record.symbol, timeframe: record.timeframe, snapshot: chartLayout, savedLayoutId: record.id, label: record.name, remountCount: t.remountCount + 1 }
499
- : t,
500
- ));
501
- }, [activeTabId, applyConfig, layouts]);
502
-
503
- // ── Layout rename ──────────────────────────────────────────────────────────
504
-
505
- const handleRenameLayout = useCallback(async (newName: string) => {
506
- if (!activeTab.savedLayoutId) throw new Error('Layout is not saved yet');
507
- await layouts.update(activeTab.savedLayoutId, { name: newName });
508
- setTabs((prev) => prev.map((t) => (t.id === activeTabId ? { ...t, label: newName } : t)));
509
- }, [activeTab.savedLayoutId, activeTabId, layouts]);
510
-
511
- // ── Layout copy ────────────────────────────────────────────────────────────
512
-
513
- const handleCopyLayout = useCallback(async (copyName: string) => {
514
- const snap = captureSnapshot();
515
- if (!snap) throw new Error('No chart snapshot available');
516
- const record = await layouts.create({
517
- name: copyName, symbol: activeTab.symbol, timeframe: activeTab.timeframe, config: buildConfig(snap),
518
- });
519
- const chartLayout = applyConfig(record.config);
520
- const newTab: Tab = {
521
- id: uid(), label: record.name, symbol: record.symbol, timeframe: record.timeframe,
522
- snapshot: chartLayout, savedLayoutId: record.id, remountCount: 0,
523
- };
524
- setTabs((prev) => [
525
- ...prev.map((t): Tab => (t.id === activeTabId ? { ...t, snapshot: snap } : t)),
526
- newTab,
527
- ]);
528
- setActiveTabId(newTab.id);
529
- }, [activeTab, activeTabId, applyConfig, buildConfig, captureSnapshot, layouts]);
530
-
531
- // ── Layout delete ──────────────────────────────────────────────────────────
532
-
533
- const handleDeleteLayout = useCallback(async (layoutId: string) => {
534
- await layouts.remove(layoutId);
535
- setTabs((prev) => prev.map((t) => {
536
- if (t.savedLayoutId !== layoutId) return t;
537
- const { savedLayoutId: _removed, ...rest } = t;
538
- return rest as Tab;
539
- }));
540
- }, [layouts]);
541
-
542
- // ── Open layout in new tab ─────────────────────────────────────────────────
543
-
544
- const handleOpenLayoutInNewTab = useCallback(async (layoutId: string) => {
545
- const snap = captureSnapshot();
546
- const record = await layouts.get(layoutId);
547
- const chartLayout = applyConfig(record.config);
548
- const newTab: Tab = {
549
- id: uid(), label: record.name, symbol: record.symbol, timeframe: record.timeframe,
550
- snapshot: chartLayout, savedLayoutId: record.id, remountCount: 0,
551
- };
552
- setTabs((prev) => [
553
- ...prev.map((t): Tab => (t.id === activeTabId && snap ? { ...t, snapshot: snap } : t)),
554
- newTab,
555
- ]);
556
- setActiveTabId(newTab.id);
557
- }, [activeTabId, applyConfig, captureSnapshot, layouts]);
558
-
559
- // ── Auto-save ──────────────────────────────────────────────────────────────
560
-
561
- const handleToggleAutoSave = useCallback((enabled: boolean) => { setAutoSave(enabled); }, []);
562
-
563
- const autoSaveFnRef = useRef<() => void>(() => {});
564
- autoSaveFnRef.current = () => {
565
- if (!autoSave || !activeTab.savedLayoutId) return;
566
- const snap = chartHandleRef.current?.getLayoutSnapshot();
567
- if (!snap) return;
568
- layouts.update(activeTab.savedLayoutId, {
569
- name: activeTab.label, symbol: activeTab.symbol, timeframe: activeTab.timeframe, config: buildConfig(snap),
570
- }).catch(() => { /* silent */ });
571
- };
572
-
573
- useEffect(() => {
574
- if (!autoSave || !activeTab.savedLayoutId) return;
575
- const id = setInterval(() => autoSaveFnRef.current(), 5_000);
576
- return () => clearInterval(id);
577
- // eslint-disable-next-line react-hooks/exhaustive-deps
578
- }, [autoSave, activeTab.savedLayoutId, activeTab.id]);
579
-
580
- // ── Chart slot assembly ────────────────────────────────────────────────────
581
-
582
- const chartSlots = (
583
- <>
584
- {tabs.map((tab) => (
585
- <div
586
- key={tab.id}
587
- style={
588
- tab.id === activeTabId
589
- ? { position: 'absolute', inset: 0, display: 'flex' }
590
- : { position: 'absolute', inset: 0, visibility: 'hidden', pointerEvents: 'none' }
591
- }
592
- >
593
- <ChartCanvas
594
- key={`${tab.id}-${tab.remountCount}`}
595
- ref={(handle) => {
596
- if (handle) {
597
- chartInstancesRef.current.set(tab.id, handle);
598
- if (tab.id === activeTabId) chartHandleRef.current = handle;
599
- } else {
600
- chartInstancesRef.current.delete(tab.id);
601
- }
602
- }}
603
- symbol={tab.symbol}
604
- timeframe={tab.timeframe}
605
- theme={theme}
606
- timezone={timezone}
607
- datafeed={datafeed}
608
- initialIndicators={tab.snapshot ? [] : DEFAULT_INDICATORS}
609
- {...(tab.snapshot ? { layoutToRestore: tab.snapshot } : {})}
610
- {...(getExchangeLogoUrl ? { getExchangeLogoUrl } : {})}
611
- {...(tradingBridge ? { tradingBridge } : {})}
612
- onIndicatorsChanged={() => setChartDirty((d) => d + 1)}
613
- />
614
- </div>
615
- ))}
616
- </>
617
- );
618
-
619
- // ── Drawer assembly ────────────────────────────────────────────────────────
620
-
621
- const drawers = (
622
- <>
623
- {scriptDrawerOpen && renderScriptDrawer?.({
624
- onClose: () => setScriptDrawerOpen(false),
625
- onAddIndicator: handleAddIndicator,
626
- })}
627
- {watchlistOpen && renderWatchlistDrawer?.({
628
- onClose: () => setWatchlistOpen(false),
629
- onSelectSymbol: (sym) => { handleSymbolChange(sym); },
630
- })}
631
- </>
632
- );
633
-
634
- // ── Render ─────────────────────────────────────────────────────────────────
635
-
636
- return (
637
- <div ref={containerRef} style={{ height: '100%' }}>
638
- {tradeDrawerOpen && capabilities.renderManagedTradingControls && renderTradeDrawer?.({
639
- onClose: () => setTradeDrawerOpen(false),
640
- })}
641
- <ChartWorkspace
642
- // TabBar
643
- tabs={tabs.map((t) => ({ id: t.id, label: t.label, isSaved: !!t.savedLayoutId }))}
644
- activeTabId={activeTabId}
645
- onSelectTab={selectTab}
646
- onAddTab={addTab}
647
- onCloseTab={closeTab}
648
- onRenameTab={renameTab}
649
- // TopToolbar
650
- symbol={activeTab.symbol}
651
- timeframe={activeTab.timeframe}
652
- theme={theme}
653
- customTimeframes={customTimeframes}
654
- favoriteTfs={favoriteTfs}
655
- onSymbolChange={handleSymbolChange}
656
- onTimeframeChange={handleTimeframeChange}
657
- onAddCustomTimeframe={handleAddCustomTimeframe}
658
- onFavoriteTfsChange={setFavoriteTfs}
659
- onAddIndicator={handleAddIndicator}
660
- onToggleTheme={toggleTheme}
661
- onCopyScreenshot={handleCopyScreenshot}
662
- onDownloadScreenshot={handleDownloadScreenshot}
663
- onFullscreen={handleFullscreen}
664
- isFullscreen={isFullscreen}
665
- {...(activeTab.savedLayoutId ? { currentLayoutName: activeTab.label, currentLayoutId: activeTab.savedLayoutId } : {})}
666
- autoSave={autoSave}
667
- onFetchLayouts={() => layouts.list()}
668
- onSaveLayout={handleSaveLayout}
669
- onLoadLayout={handleLoadLayout}
670
- onRenameLayout={handleRenameLayout}
671
- onCopyLayout={handleCopyLayout}
672
- onToggleAutoSave={handleToggleAutoSave}
673
- onDeleteLayout={handleDeleteLayout}
674
- onOpenLayoutInNewTab={handleOpenLayoutInNewTab}
675
- symbolResolver={symbolResolver}
676
- showTradeButton={capabilities.renderManagedTradingControls}
677
- tradeDrawerOpen={tradeDrawerOpen}
678
- onToggleTradeDrawer={() => setTradeDrawerOpen((o) => !o)}
679
- // RightToolbar
680
- watchlistOpen={watchlistOpen}
681
- onToggleWatchlist={() => setWatchlistOpen((o) => !o)}
682
- // BottomToolbar
683
- activeSymbol={activeTab.symbol}
684
- timezone={timezone}
685
- onTimezoneChange={setTimezone}
686
- session={session}
687
- onSessionChange={setSession}
688
- scriptDrawerOpen={scriptDrawerOpen}
689
- onToggleScriptDrawer={() => setScriptDrawerOpen((o) => !o)}
690
- // State
691
- isLicensed={isLicensed}
692
- wsLoaded={wsLoaded}
693
- // Slots
694
- chartSlots={chartSlots}
695
- drawers={drawers}
696
- />
697
- </div>
698
- );
699
- });