@forgecharts/sdk 1.1.27 → 1.1.28
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.
- package/dist/__tests__/backwardCompatibility.test.d.ts +14 -0
- package/dist/__tests__/backwardCompatibility.test.d.ts.map +1 -0
- package/dist/__tests__/backwardCompatibility.test.js +159 -0
- package/dist/__tests__/backwardCompatibility.test.js.map +1 -0
- package/dist/__tests__/candleInvariant.test.d.ts +20 -0
- package/dist/__tests__/candleInvariant.test.d.ts.map +1 -0
- package/dist/__tests__/candleInvariant.test.js +415 -0
- package/dist/__tests__/candleInvariant.test.js.map +1 -0
- package/dist/__tests__/public-api-surface.d.ts +13 -0
- package/dist/__tests__/public-api-surface.d.ts.map +1 -0
- package/{src/__tests__/public-api-surface.ts → dist/__tests__/public-api-surface.js} +4 -42
- package/dist/__tests__/public-api-surface.js.map +1 -0
- package/dist/__tests__/timeframeBoundary.test.d.ts +17 -0
- package/dist/__tests__/timeframeBoundary.test.d.ts.map +1 -0
- package/dist/__tests__/timeframeBoundary.test.js +452 -0
- package/dist/__tests__/timeframeBoundary.test.js.map +1 -0
- package/dist/api/DrawingManager.d.ts +20 -0
- package/dist/api/DrawingManager.d.ts.map +1 -0
- package/dist/api/DrawingManager.js +190 -0
- package/dist/api/DrawingManager.js.map +1 -0
- package/dist/api/EventBus.d.ts +19 -0
- package/dist/api/EventBus.d.ts.map +1 -0
- package/dist/api/EventBus.js +44 -0
- package/dist/api/EventBus.js.map +1 -0
- package/dist/api/IndicatorDAG.d.ts +85 -0
- package/dist/api/IndicatorDAG.d.ts.map +1 -0
- package/dist/api/IndicatorDAG.js +316 -0
- package/dist/api/IndicatorDAG.js.map +1 -0
- package/dist/api/IndicatorRegistry.d.ts +22 -0
- package/dist/api/IndicatorRegistry.d.ts.map +1 -0
- package/dist/api/IndicatorRegistry.js +39 -0
- package/dist/api/IndicatorRegistry.js.map +1 -0
- package/dist/api/LayoutManager.d.ts +30 -0
- package/dist/api/LayoutManager.d.ts.map +1 -0
- package/dist/api/LayoutManager.js +51 -0
- package/dist/api/LayoutManager.js.map +1 -0
- package/dist/api/PaneManager.d.ts +42 -0
- package/dist/api/PaneManager.d.ts.map +1 -0
- package/dist/api/PaneManager.js +119 -0
- package/dist/api/PaneManager.js.map +1 -0
- package/dist/api/ReferenceAPI.d.ts +78 -0
- package/dist/api/ReferenceAPI.d.ts.map +1 -0
- package/dist/api/ReferenceAPI.js +153 -0
- package/dist/api/ReferenceAPI.js.map +1 -0
- package/dist/api/TChart.d.ts +345 -0
- package/dist/api/TChart.d.ts.map +1 -0
- package/dist/api/TChart.js +765 -0
- package/dist/api/TChart.js.map +1 -0
- package/{src/api/createChart.ts → dist/api/createChart.d.ts} +2 -7
- package/dist/api/createChart.d.ts.map +1 -0
- package/dist/api/createChart.js +42 -0
- package/dist/api/createChart.js.map +1 -0
- package/dist/api/drawing tools/fib gann menu/fibRetracement.d.ts +4 -0
- package/dist/api/drawing tools/fib gann menu/fibRetracement.d.ts.map +1 -0
- package/dist/api/drawing tools/fib gann menu/fibRetracement.js +22 -0
- package/dist/api/drawing tools/fib gann menu/fibRetracement.js.map +1 -0
- package/dist/api/drawing tools/lines menu/crossLine.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/crossLine.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/crossLine.js +16 -0
- package/dist/api/drawing tools/lines menu/crossLine.js.map +1 -0
- package/dist/api/drawing tools/lines menu/disjointChannel.d.ts +17 -0
- package/dist/api/drawing tools/lines menu/disjointChannel.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/disjointChannel.js +59 -0
- package/dist/api/drawing tools/lines menu/disjointChannel.js.map +1 -0
- package/dist/api/drawing tools/lines menu/extendedLine.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/extendedLine.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/extendedLine.js +17 -0
- package/dist/api/drawing tools/lines menu/extendedLine.js.map +1 -0
- package/dist/api/drawing tools/lines menu/flatTopBottom.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/flatTopBottom.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/flatTopBottom.js +41 -0
- package/dist/api/drawing tools/lines menu/flatTopBottom.js.map +1 -0
- package/dist/api/drawing tools/lines menu/horizontal.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/horizontal.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/horizontal.js +19 -0
- package/dist/api/drawing tools/lines menu/horizontal.js.map +1 -0
- package/dist/api/drawing tools/lines menu/horizontalRay.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/horizontalRay.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/horizontalRay.js +20 -0
- package/dist/api/drawing tools/lines menu/horizontalRay.js.map +1 -0
- package/dist/api/drawing tools/lines menu/infoLine.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/infoLine.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/infoLine.js +107 -0
- package/dist/api/drawing tools/lines menu/infoLine.js.map +1 -0
- package/dist/api/drawing tools/lines menu/insidePitchfork.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/insidePitchfork.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/insidePitchfork.js +31 -0
- package/dist/api/drawing tools/lines menu/insidePitchfork.js.map +1 -0
- package/dist/api/drawing tools/lines menu/modifiedSchiffPitchfork.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/modifiedSchiffPitchfork.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/modifiedSchiffPitchfork.js +15 -0
- package/dist/api/drawing tools/lines menu/modifiedSchiffPitchfork.js.map +1 -0
- package/dist/api/drawing tools/lines menu/parallelChannel.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/parallelChannel.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/parallelChannel.js +43 -0
- package/dist/api/drawing tools/lines menu/parallelChannel.js.map +1 -0
- package/dist/api/drawing tools/lines menu/pitchfork.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/pitchfork.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/pitchfork.js +12 -0
- package/dist/api/drawing tools/lines menu/pitchfork.js.map +1 -0
- package/dist/api/drawing tools/lines menu/ray.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/ray.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/ray.js +23 -0
- package/dist/api/drawing tools/lines menu/ray.js.map +1 -0
- package/dist/api/drawing tools/lines menu/regressionTrend.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/regressionTrend.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/regressionTrend.js +127 -0
- package/dist/api/drawing tools/lines menu/regressionTrend.js.map +1 -0
- package/dist/api/drawing tools/lines menu/schiffPitchfork.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/schiffPitchfork.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/schiffPitchfork.js +15 -0
- package/dist/api/drawing tools/lines menu/schiffPitchfork.js.map +1 -0
- package/dist/api/drawing tools/lines menu/trendAngle.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/trendAngle.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/trendAngle.js +51 -0
- package/dist/api/drawing tools/lines menu/trendAngle.js.map +1 -0
- package/dist/api/drawing tools/lines menu/trendline.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/trendline.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/trendline.js +11 -0
- package/dist/api/drawing tools/lines menu/trendline.js.map +1 -0
- package/dist/api/drawing tools/lines menu/vertical.d.ts +4 -0
- package/dist/api/drawing tools/lines menu/vertical.d.ts.map +1 -0
- package/dist/api/drawing tools/lines menu/vertical.js +11 -0
- package/dist/api/drawing tools/lines menu/vertical.js.map +1 -0
- package/dist/api/drawing tools/pointers menu/crosshair.d.ts +16 -0
- package/dist/api/drawing tools/pointers menu/crosshair.d.ts.map +1 -0
- package/{src/api/drawing tools/pointers menu/crosshair.ts → dist/api/drawing tools/pointers menu/crosshair.js } +6 -7
- package/dist/api/drawing tools/pointers menu/crosshair.js.map +1 -0
- package/dist/api/drawing tools/pointers menu/cursor.d.ts +15 -0
- package/dist/api/drawing tools/pointers menu/cursor.d.ts.map +1 -0
- package/{src/api/drawing tools/pointers menu/cursor.ts → dist/api/drawing tools/pointers menu/cursor.js } +7 -8
- package/dist/api/drawing tools/pointers menu/cursor.js.map +1 -0
- package/dist/api/drawing tools/pointers menu/demonstration.d.ts +30 -0
- package/dist/api/drawing tools/pointers menu/demonstration.d.ts.map +1 -0
- package/{src/api/drawing tools/pointers menu/demonstration.ts → dist/api/drawing tools/pointers menu/demonstration.js } +9 -14
- package/dist/api/drawing tools/pointers menu/demonstration.js.map +1 -0
- package/dist/api/drawing tools/pointers menu/dot.d.ts +23 -0
- package/dist/api/drawing tools/pointers menu/dot.d.ts.map +1 -0
- package/{src/api/drawing tools/pointers menu/dot.ts → dist/api/drawing tools/pointers menu/dot.js } +8 -11
- package/dist/api/drawing tools/pointers menu/dot.js.map +1 -0
- package/dist/api/drawing tools/shapes menu/rectangle.d.ts +4 -0
- package/dist/api/drawing tools/shapes menu/rectangle.d.ts.map +1 -0
- package/dist/api/drawing tools/shapes menu/rectangle.js +19 -0
- package/dist/api/drawing tools/shapes menu/rectangle.js.map +1 -0
- package/dist/api/drawing tools/shapes menu/text.d.ts +4 -0
- package/dist/api/drawing tools/shapes menu/text.d.ts.map +1 -0
- package/dist/api/drawing tools/shapes menu/text.js +25 -0
- package/dist/api/drawing tools/shapes menu/text.js.map +1 -0
- package/dist/api/drawingUtils.d.ts +22 -0
- package/dist/api/drawingUtils.d.ts.map +1 -0
- package/dist/api/drawingUtils.js +83 -0
- package/dist/api/drawingUtils.js.map +1 -0
- package/dist/core/CanvasLayer.d.ts +26 -0
- package/dist/core/CanvasLayer.d.ts.map +1 -0
- package/dist/core/CanvasLayer.js +56 -0
- package/dist/core/CanvasLayer.js.map +1 -0
- package/dist/core/Chart.d.ts +164 -0
- package/dist/core/Chart.d.ts.map +1 -0
- package/dist/core/Chart.js +839 -0
- package/dist/core/Chart.js.map +1 -0
- package/dist/core/CoordTransform.d.ts +168 -0
- package/dist/core/CoordTransform.d.ts.map +1 -0
- package/dist/core/CoordTransform.js +224 -0
- package/dist/core/CoordTransform.js.map +1 -0
- package/dist/core/Crosshair.d.ts +30 -0
- package/dist/core/Crosshair.d.ts.map +1 -0
- package/dist/core/Crosshair.js +186 -0
- package/dist/core/Crosshair.js.map +1 -0
- package/dist/core/IndicatorEngine.d.ts +51 -0
- package/dist/core/IndicatorEngine.d.ts.map +1 -0
- package/dist/core/IndicatorEngine.js +181 -0
- package/dist/core/IndicatorEngine.js.map +1 -0
- package/dist/core/InteractionManager.d.ts +197 -0
- package/dist/core/InteractionManager.d.ts.map +1 -0
- package/dist/core/InteractionManager.js +698 -0
- package/dist/core/InteractionManager.js.map +1 -0
- package/dist/core/PriceScale.d.ts +27 -0
- package/dist/core/PriceScale.d.ts.map +1 -0
- package/dist/core/PriceScale.js +113 -0
- package/dist/core/PriceScale.js.map +1 -0
- package/dist/core/Series.d.ts +40 -0
- package/dist/core/Series.d.ts.map +1 -0
- package/dist/core/Series.js +114 -0
- package/dist/core/Series.js.map +1 -0
- package/dist/core/TimeScale.d.ts +43 -0
- package/dist/core/TimeScale.d.ts.map +1 -0
- package/dist/core/TimeScale.js +150 -0
- package/dist/core/TimeScale.js.map +1 -0
- package/dist/datafeed/DatafeedConnector.d.ts +89 -0
- package/dist/datafeed/DatafeedConnector.d.ts.map +1 -0
- package/dist/datafeed/DatafeedConnector.js +268 -0
- package/dist/datafeed/DatafeedConnector.js.map +1 -0
- package/dist/engine/CandleEngine.d.ts +207 -0
- package/dist/engine/CandleEngine.d.ts.map +1 -0
- package/dist/engine/CandleEngine.js +318 -0
- package/dist/engine/CandleEngine.js.map +1 -0
- package/dist/engine/__tests__/CandleEngine.test.d.ts +2 -0
- package/dist/engine/__tests__/CandleEngine.test.d.ts.map +1 -0
- package/dist/engine/__tests__/CandleEngine.test.js +300 -0
- package/dist/engine/__tests__/CandleEngine.test.js.map +1 -0
- package/dist/engine/candleInvariants.d.ts +66 -0
- package/dist/engine/candleInvariants.d.ts.map +1 -0
- package/dist/engine/candleInvariants.js +134 -0
- package/dist/engine/candleInvariants.js.map +1 -0
- package/{src/engine/mergeUtils.ts → dist/engine/mergeUtils.d.ts} +15 -52
- package/dist/engine/mergeUtils.d.ts.map +1 -0
- package/dist/engine/mergeUtils.js +64 -0
- package/dist/engine/mergeUtils.js.map +1 -0
- package/dist/engine/timeframeUtils.d.ts +80 -0
- package/dist/engine/timeframeUtils.d.ts.map +1 -0
- package/{src/engine/timeframeUtils.ts → dist/engine/timeframeUtils.js} +30 -48
- package/dist/engine/timeframeUtils.js.map +1 -0
- package/dist/index.d.ts +40 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +49 -0
- package/dist/index.js.map +1 -0
- package/dist/internal.d.ts +29 -0
- package/dist/internal.d.ts.map +1 -0
- package/{src/internal.ts → dist/internal.js} +1 -10
- package/dist/internal.js.map +1 -0
- package/dist/licensing/ChartRuntimeResolver.d.ts +233 -0
- package/dist/licensing/ChartRuntimeResolver.d.ts.map +1 -0
- package/dist/licensing/ChartRuntimeResolver.js +310 -0
- package/dist/licensing/ChartRuntimeResolver.js.map +1 -0
- package/dist/licensing/LicenseManager.d.ts +55 -0
- package/dist/licensing/LicenseManager.d.ts.map +1 -0
- package/dist/licensing/LicenseManager.js +114 -0
- package/dist/licensing/LicenseManager.js.map +1 -0
- package/dist/licensing/__tests__/ChartRuntimeResolver.test.d.ts +13 -0
- package/dist/licensing/__tests__/ChartRuntimeResolver.test.d.ts.map +1 -0
- package/dist/licensing/__tests__/ChartRuntimeResolver.test.js +177 -0
- package/dist/licensing/__tests__/ChartRuntimeResolver.test.js.map +1 -0
- package/dist/licensing/__tests__/LicenseManager.test.d.ts +12 -0
- package/dist/licensing/__tests__/LicenseManager.test.d.ts.map +1 -0
- package/dist/licensing/__tests__/LicenseManager.test.js +153 -0
- package/dist/licensing/__tests__/LicenseManager.test.js.map +1 -0
- package/dist/licensing/licenseTypes.d.ts +18 -0
- package/dist/licensing/licenseTypes.d.ts.map +1 -0
- package/dist/licensing/licenseTypes.js +2 -0
- package/dist/licensing/licenseTypes.js.map +1 -0
- package/dist/pine/PineCompiler.d.ts +35 -0
- package/dist/pine/PineCompiler.d.ts.map +1 -0
- package/dist/pine/PineCompiler.js +44 -0
- package/dist/pine/PineCompiler.js.map +1 -0
- package/dist/pine/diagnostics.d.ts +20 -0
- package/dist/pine/diagnostics.d.ts.map +1 -0
- package/dist/pine/diagnostics.js +11 -0
- package/dist/pine/diagnostics.js.map +1 -0
- package/{src/pine/index.ts → dist/pine/index.d.ts} +4 -3
- package/dist/pine/index.d.ts.map +1 -0
- package/dist/pine/index.js +5 -0
- package/dist/pine/index.js.map +1 -0
- package/dist/pine/pine-ast.d.ts +142 -0
- package/dist/pine/pine-ast.d.ts.map +1 -0
- package/dist/pine/pine-ast.js +19 -0
- package/dist/pine/pine-ast.js.map +1 -0
- package/dist/pine/pine-lexer.d.ts +41 -0
- package/dist/pine/pine-lexer.d.ts.map +1 -0
- package/dist/pine/pine-lexer.js +249 -0
- package/dist/pine/pine-lexer.js.map +1 -0
- package/dist/pine/pine-parser.d.ts +51 -0
- package/dist/pine/pine-parser.d.ts.map +1 -0
- package/dist/pine/pine-parser.js +416 -0
- package/dist/pine/pine-parser.js.map +1 -0
- package/dist/pine/pine-transpiler.d.ts +33 -0
- package/dist/pine/pine-transpiler.d.ts.map +1 -0
- package/dist/pine/pine-transpiler.js +260 -0
- package/dist/pine/pine-transpiler.js.map +1 -0
- package/dist/pixi/LayerName.d.ts +18 -0
- package/dist/pixi/LayerName.d.ts.map +1 -0
- package/dist/pixi/LayerName.js +35 -0
- package/dist/pixi/LayerName.js.map +1 -0
- package/dist/pixi/PixiCandlestickRenderer.d.ts +23 -0
- package/dist/pixi/PixiCandlestickRenderer.d.ts.map +1 -0
- package/dist/pixi/PixiCandlestickRenderer.js +107 -0
- package/dist/pixi/PixiCandlestickRenderer.js.map +1 -0
- package/dist/pixi/PixiChart.d.ts +72 -0
- package/dist/pixi/PixiChart.d.ts.map +1 -0
- package/dist/pixi/PixiChart.js +367 -0
- package/dist/pixi/PixiChart.js.map +1 -0
- package/dist/pixi/PixiCrosshairRenderer.d.ts +29 -0
- package/dist/pixi/PixiCrosshairRenderer.d.ts.map +1 -0
- package/dist/pixi/PixiCrosshairRenderer.js +110 -0
- package/dist/pixi/PixiCrosshairRenderer.js.map +1 -0
- package/dist/pixi/PixiDrawingRenderer.d.ts +17 -0
- package/dist/pixi/PixiDrawingRenderer.d.ts.map +1 -0
- package/dist/pixi/PixiDrawingRenderer.js +111 -0
- package/dist/pixi/PixiDrawingRenderer.js.map +1 -0
- package/dist/pixi/PixiGridRenderer.d.ts +22 -0
- package/dist/pixi/PixiGridRenderer.d.ts.map +1 -0
- package/dist/pixi/PixiGridRenderer.js +114 -0
- package/dist/pixi/PixiGridRenderer.js.map +1 -0
- package/dist/pixi/PixiLayerManager.d.ts +56 -0
- package/dist/pixi/PixiLayerManager.d.ts.map +1 -0
- package/dist/pixi/PixiLayerManager.js +92 -0
- package/dist/pixi/PixiLayerManager.js.map +1 -0
- package/dist/react/canvas/ChartCanvas.d.ts +85 -0
- package/dist/react/canvas/ChartCanvas.d.ts.map +1 -0
- package/dist/react/canvas/ChartCanvas.js +604 -0
- package/dist/react/canvas/ChartCanvas.js.map +1 -0
- package/dist/react/canvas/ChartContextMenu.d.ts +18 -0
- package/dist/react/canvas/ChartContextMenu.d.ts.map +1 -0
- package/dist/react/canvas/ChartContextMenu.js +5 -0
- package/dist/react/canvas/ChartContextMenu.js.map +1 -0
- package/dist/react/canvas/ChartSettingsDialog.d.ts +25 -0
- package/dist/react/canvas/ChartSettingsDialog.d.ts.map +1 -0
- package/dist/react/canvas/ChartSettingsDialog.js +28 -0
- package/dist/react/canvas/ChartSettingsDialog.js.map +1 -0
- package/dist/react/canvas/IndicatorLabel.d.ts +21 -0
- package/dist/react/canvas/IndicatorLabel.d.ts.map +1 -0
- package/dist/react/canvas/IndicatorLabel.js +196 -0
- package/dist/react/canvas/IndicatorLabel.js.map +1 -0
- package/dist/react/canvas/IndicatorPane.d.ts +32 -0
- package/dist/react/canvas/IndicatorPane.d.ts.map +1 -0
- package/dist/react/canvas/IndicatorPane.js +395 -0
- package/dist/react/canvas/IndicatorPane.js.map +1 -0
- package/dist/react/canvas/PointerOverlay.d.ts +23 -0
- package/dist/react/canvas/PointerOverlay.d.ts.map +1 -0
- package/dist/react/canvas/PointerOverlay.js +61 -0
- package/dist/react/canvas/PointerOverlay.js.map +1 -0
- package/dist/react/canvas/toolbars/LeftToolbar.d.ts +19 -0
- package/dist/react/canvas/toolbars/LeftToolbar.d.ts.map +1 -0
- package/dist/react/canvas/toolbars/LeftToolbar.js +407 -0
- package/dist/react/canvas/toolbars/LeftToolbar.js.map +1 -0
- package/dist/react/hooks/useChartCapabilities.d.ts +21 -0
- package/dist/react/hooks/useChartCapabilities.d.ts.map +1 -0
- package/dist/react/hooks/useChartCapabilities.js +66 -0
- package/dist/react/hooks/useChartCapabilities.js.map +1 -0
- package/dist/react/index.d.ts +30 -0
- package/dist/react/index.d.ts.map +1 -0
- package/{src/react/index.ts → dist/react/index.js} +1 -24
- package/dist/react/index.js.map +1 -0
- package/dist/react/internal.d.ts +38 -0
- package/dist/react/internal.d.ts.map +1 -0
- package/{src/react/internal.ts → dist/react/internal.js} +1 -19
- package/dist/react/internal.js.map +1 -0
- package/dist/react/shell/ManagedAppShell.d.ts +91 -0
- package/dist/react/shell/ManagedAppShell.d.ts.map +1 -0
- package/dist/react/shell/ManagedAppShell.js +440 -0
- package/dist/react/shell/ManagedAppShell.js.map +1 -0
- package/dist/react/trading/TradingBridge.d.ts +86 -0
- package/dist/react/trading/TradingBridge.d.ts.map +1 -0
- package/dist/react/trading/TradingBridge.js +73 -0
- package/dist/react/trading/TradingBridge.js.map +1 -0
- package/dist/react/workspace/ChartWorkspace.d.ts +73 -0
- package/dist/react/workspace/ChartWorkspace.d.ts.map +1 -0
- package/dist/react/workspace/ChartWorkspace.js +42 -0
- package/dist/react/workspace/ChartWorkspace.js.map +1 -0
- package/dist/react/workspace/FloatingPanel.d.ts +18 -0
- package/dist/react/workspace/FloatingPanel.d.ts.map +1 -0
- package/dist/react/workspace/FloatingPanel.js +82 -0
- package/dist/react/workspace/FloatingPanel.js.map +1 -0
- package/dist/react/workspace/IndicatorsDialog.d.ts +8 -0
- package/dist/react/workspace/IndicatorsDialog.d.ts.map +1 -0
- package/dist/react/workspace/IndicatorsDialog.js +121 -0
- package/dist/react/workspace/IndicatorsDialog.js.map +1 -0
- package/dist/react/workspace/LayoutMenu.d.ts +33 -0
- package/dist/react/workspace/LayoutMenu.d.ts.map +1 -0
- package/dist/react/workspace/LayoutMenu.js +113 -0
- package/dist/react/workspace/LayoutMenu.js.map +1 -0
- package/dist/react/workspace/SymbolSearchDialog.d.ts +10 -0
- package/dist/react/workspace/SymbolSearchDialog.d.ts.map +1 -0
- package/dist/react/workspace/SymbolSearchDialog.js +245 -0
- package/dist/react/workspace/SymbolSearchDialog.js.map +1 -0
- package/dist/react/workspace/TabBar.d.ts +17 -0
- package/dist/react/workspace/TabBar.d.ts.map +1 -0
- package/dist/react/workspace/TabBar.js +29 -0
- package/dist/react/workspace/TabBar.js.map +1 -0
- package/dist/react/workspace/toolbars/BottomToolbar.d.ts +19 -0
- package/dist/react/workspace/toolbars/BottomToolbar.d.ts.map +1 -0
- package/dist/react/workspace/toolbars/BottomToolbar.js +236 -0
- package/dist/react/workspace/toolbars/BottomToolbar.js.map +1 -0
- package/dist/react/workspace/toolbars/RightToolbar.d.ts +8 -0
- package/dist/react/workspace/toolbars/RightToolbar.d.ts.map +1 -0
- package/dist/react/workspace/toolbars/RightToolbar.js +18 -0
- package/dist/react/workspace/toolbars/RightToolbar.js.map +1 -0
- package/dist/react/workspace/toolbars/TopToolbar.d.ts +41 -0
- package/dist/react/workspace/toolbars/TopToolbar.d.ts.map +1 -0
- package/dist/react/workspace/toolbars/TopToolbar.js +82 -0
- package/dist/react/workspace/toolbars/TopToolbar.js.map +1 -0
- package/dist/renderers/CandlestickRenderer.d.ts +13 -0
- package/dist/renderers/CandlestickRenderer.d.ts.map +1 -0
- package/dist/renderers/CandlestickRenderer.js +98 -0
- package/dist/renderers/CandlestickRenderer.js.map +1 -0
- package/dist/renderers/HistogramRenderer.d.ts +11 -0
- package/dist/renderers/HistogramRenderer.d.ts.map +1 -0
- package/dist/renderers/HistogramRenderer.js +50 -0
- package/dist/renderers/HistogramRenderer.js.map +1 -0
- package/dist/renderers/LineRenderer.d.ts +12 -0
- package/dist/renderers/LineRenderer.d.ts.map +1 -0
- package/dist/renderers/LineRenderer.js +64 -0
- package/dist/renderers/LineRenderer.js.map +1 -0
- package/dist/theme/colors.d.ts +4 -0
- package/dist/theme/colors.d.ts.map +1 -0
- package/dist/theme/colors.js +19 -0
- package/dist/theme/colors.js.map +1 -0
- package/dist/tools/barDivergenceCheck.d.ts +120 -0
- package/dist/tools/barDivergenceCheck.d.ts.map +1 -0
- package/dist/tools/barDivergenceCheck.js +200 -0
- package/dist/tools/barDivergenceCheck.js.map +1 -0
- package/dist/trading/TradingOverlayStore.d.ts +86 -0
- package/dist/trading/TradingOverlayStore.d.ts.map +1 -0
- package/dist/trading/TradingOverlayStore.js +139 -0
- package/dist/trading/TradingOverlayStore.js.map +1 -0
- package/dist/trading/UnmanagedIngestion.d.ts +91 -0
- package/dist/trading/UnmanagedIngestion.d.ts.map +1 -0
- package/dist/trading/UnmanagedIngestion.js +114 -0
- package/dist/trading/UnmanagedIngestion.js.map +1 -0
- package/dist/trading/__tests__/ManagedTradingController.test.d.ts +18 -0
- package/dist/trading/__tests__/ManagedTradingController.test.d.ts.map +1 -0
- package/dist/trading/__tests__/ManagedTradingController.test.js +271 -0
- package/dist/trading/__tests__/ManagedTradingController.test.js.map +1 -0
- package/dist/trading/__tests__/TradingOverlayStore.test.d.ts +16 -0
- package/dist/trading/__tests__/TradingOverlayStore.test.d.ts.map +1 -0
- package/dist/trading/__tests__/TradingOverlayStore.test.js +267 -0
- package/dist/trading/__tests__/TradingOverlayStore.test.js.map +1 -0
- package/dist/trading/__tests__/UnmanagedIngestion.test.d.ts +16 -0
- package/dist/trading/__tests__/UnmanagedIngestion.test.d.ts.map +1 -0
- package/dist/trading/__tests__/UnmanagedIngestion.test.js +170 -0
- package/dist/trading/__tests__/UnmanagedIngestion.test.js.map +1 -0
- package/dist/trading/managed/ManagedTradingController.d.ts +110 -0
- package/dist/trading/managed/ManagedTradingController.d.ts.map +1 -0
- package/dist/trading/managed/ManagedTradingController.js +247 -0
- package/dist/trading/managed/ManagedTradingController.js.map +1 -0
- package/dist/trading/managed/managedCapabilities.d.ts +45 -0
- package/dist/trading/managed/managedCapabilities.d.ts.map +1 -0
- package/dist/trading/managed/managedCapabilities.js +79 -0
- package/dist/trading/managed/managedCapabilities.js.map +1 -0
- package/dist/trading/managed/managedTypes.d.ts +122 -0
- package/dist/trading/managed/managedTypes.d.ts.map +1 -0
- package/dist/trading/managed/managedTypes.js +13 -0
- package/dist/trading/managed/managedTypes.js.map +1 -0
- package/dist/trading/tradingTypes.d.ts +89 -0
- package/dist/trading/tradingTypes.d.ts.map +1 -0
- package/dist/trading/tradingTypes.js +8 -0
- package/dist/trading/tradingTypes.js.map +1 -0
- package/dist/tscript/TScriptIndicator.d.ts +41 -0
- package/dist/tscript/TScriptIndicator.d.ts.map +1 -0
- package/dist/tscript/TScriptIndicator.js +47 -0
- package/dist/tscript/TScriptIndicator.js.map +1 -0
- package/dist/tscript/ast.d.ts +89 -0
- package/dist/tscript/ast.d.ts.map +1 -0
- package/dist/tscript/ast.js +22 -0
- package/dist/tscript/ast.js.map +1 -0
- package/dist/tscript/lexer.d.ts +36 -0
- package/dist/tscript/lexer.d.ts.map +1 -0
- package/dist/tscript/lexer.js +187 -0
- package/dist/tscript/lexer.js.map +1 -0
- package/dist/tscript/parser.d.ts +50 -0
- package/dist/tscript/parser.d.ts.map +1 -0
- package/dist/tscript/parser.js +318 -0
- package/dist/tscript/parser.js.map +1 -0
- package/dist/tscript/runtime.d.ts +123 -0
- package/dist/tscript/runtime.d.ts.map +1 -0
- package/dist/tscript/runtime.js +475 -0
- package/dist/tscript/runtime.js.map +1 -0
- package/dist/tscript/series.d.ts +49 -0
- package/dist/tscript/series.d.ts.map +1 -0
- package/dist/tscript/series.js +79 -0
- package/dist/tscript/series.js.map +1 -0
- package/dist/types/IChart.d.ts +48 -0
- package/dist/types/IChart.d.ts.map +1 -0
- package/dist/types/IChart.js +2 -0
- package/dist/types/IChart.js.map +1 -0
- package/{src/types/IRenderer.ts → dist/types/IRenderer.d.ts} +2 -8
- package/dist/types/IRenderer.d.ts.map +1 -0
- package/dist/types/IRenderer.js +2 -0
- package/dist/types/IRenderer.js.map +1 -0
- package/dist/types/ISeries.d.ts +26 -0
- package/dist/types/ISeries.d.ts.map +1 -0
- package/dist/types/ISeries.js +2 -0
- package/dist/types/ISeries.js.map +1 -0
- package/package.json +5 -1
- package/src/__tests__/backwardCompatibility.test.ts +0 -191
- package/src/__tests__/candleInvariant.test.ts +0 -500
- package/src/__tests__/timeframeBoundary.test.ts +0 -583
- package/src/api/DrawingManager.ts +0 -188
- package/src/api/EventBus.ts +0 -53
- package/src/api/IndicatorDAG.ts +0 -389
- package/src/api/IndicatorRegistry.ts +0 -47
- package/src/api/LayoutManager.ts +0 -72
- package/src/api/PaneManager.ts +0 -129
- package/src/api/ReferenceAPI.ts +0 -195
- package/src/api/TChart.ts +0 -881
- package/src/api/drawing tools/fib gann menu/fibRetracement.ts +0 -27
- package/src/api/drawing tools/lines menu/crossLine.ts +0 -21
- package/src/api/drawing tools/lines menu/disjointChannel.ts +0 -74
- package/src/api/drawing tools/lines menu/extendedLine.ts +0 -22
- package/src/api/drawing tools/lines menu/flatTopBottom.ts +0 -45
- package/src/api/drawing tools/lines menu/horizontal.ts +0 -24
- package/src/api/drawing tools/lines menu/horizontalRay.ts +0 -25
- package/src/api/drawing tools/lines menu/infoLine.ts +0 -127
- package/src/api/drawing tools/lines menu/insidePitchfork.ts +0 -21
- package/src/api/drawing tools/lines menu/modifiedSchiffPitchfork.ts +0 -18
- package/src/api/drawing tools/lines menu/parallelChannel.ts +0 -47
- package/src/api/drawing tools/lines menu/pitchfork.ts +0 -15
- package/src/api/drawing tools/lines menu/ray.ts +0 -28
- package/src/api/drawing tools/lines menu/regressionTrend.ts +0 -157
- package/src/api/drawing tools/lines menu/schiffPitchfork.ts +0 -18
- package/src/api/drawing tools/lines menu/trendAngle.ts +0 -64
- package/src/api/drawing tools/lines menu/trendline.ts +0 -16
- package/src/api/drawing tools/lines menu/vertical.ts +0 -16
- package/src/api/drawing tools/shapes menu/rectangle.ts +0 -24
- package/src/api/drawing tools/shapes menu/text.ts +0 -30
- package/src/api/drawingUtils.ts +0 -82
- package/src/core/CanvasLayer.ts +0 -77
- package/src/core/Chart.ts +0 -917
- package/src/core/CoordTransform.ts +0 -282
- package/src/core/Crosshair.ts +0 -207
- package/src/core/IndicatorEngine.ts +0 -216
- package/src/core/InteractionManager.ts +0 -899
- package/src/core/PriceScale.ts +0 -133
- package/src/core/Series.ts +0 -132
- package/src/core/TimeScale.ts +0 -175
- package/src/datafeed/DatafeedConnector.ts +0 -300
- package/src/engine/CandleEngine.ts +0 -458
- package/src/engine/__tests__/CandleEngine.test.ts +0 -402
- package/src/engine/candleInvariants.ts +0 -172
- package/src/index.ts +0 -190
- package/src/licensing/ChartRuntimeResolver.ts +0 -380
- package/src/licensing/LicenseManager.ts +0 -131
- package/src/licensing/__tests__/ChartRuntimeResolver.test.ts +0 -207
- package/src/licensing/__tests__/LicenseManager.test.ts +0 -180
- package/src/licensing/licenseTypes.ts +0 -19
- package/src/pine/PineCompiler.ts +0 -68
- package/src/pine/diagnostics.ts +0 -30
- package/src/pine/pine-ast.ts +0 -163
- package/src/pine/pine-lexer.ts +0 -265
- package/src/pine/pine-parser.ts +0 -439
- package/src/pine/pine-transpiler.ts +0 -301
- package/src/pixi/LayerName.ts +0 -35
- package/src/pixi/PixiCandlestickRenderer.ts +0 -125
- package/src/pixi/PixiChart.ts +0 -425
- package/src/pixi/PixiCrosshairRenderer.ts +0 -134
- package/src/pixi/PixiDrawingRenderer.ts +0 -121
- package/src/pixi/PixiGridRenderer.ts +0 -136
- package/src/pixi/PixiLayerManager.ts +0 -102
- package/src/react/canvas/ChartCanvas.tsx +0 -984
- package/src/react/canvas/ChartContextMenu.tsx +0 -60
- package/src/react/canvas/ChartSettingsDialog.tsx +0 -133
- package/src/react/canvas/IndicatorLabel.tsx +0 -347
- package/src/react/canvas/IndicatorPane.tsx +0 -503
- package/src/react/canvas/PointerOverlay.tsx +0 -126
- package/src/react/canvas/toolbars/LeftToolbar.tsx +0 -1096
- package/src/react/hooks/useChartCapabilities.ts +0 -76
- package/src/react/shell/ManagedAppShell.tsx +0 -699
- package/src/react/trading/TradingBridge.ts +0 -156
- package/src/react/workspace/ChartWorkspace.tsx +0 -228
- package/src/react/workspace/FloatingPanel.tsx +0 -131
- package/src/react/workspace/IndicatorsDialog.tsx +0 -246
- package/src/react/workspace/LayoutMenu.tsx +0 -345
- package/src/react/workspace/SymbolSearchDialog.tsx +0 -377
- package/src/react/workspace/TabBar.tsx +0 -87
- package/src/react/workspace/toolbars/BottomToolbar.tsx +0 -372
- package/src/react/workspace/toolbars/RightToolbar.tsx +0 -46
- package/src/react/workspace/toolbars/TopToolbar.tsx +0 -431
- package/src/renderers/CandlestickRenderer.ts +0 -130
- package/src/renderers/HistogramRenderer.ts +0 -63
- package/src/renderers/LineRenderer.ts +0 -77
- package/src/theme/colors.ts +0 -21
- package/src/tools/barDivergenceCheck.ts +0 -305
- package/src/trading/TradingOverlayStore.ts +0 -161
- package/src/trading/UnmanagedIngestion.ts +0 -156
- package/src/trading/__tests__/ManagedTradingController.test.ts +0 -338
- package/src/trading/__tests__/TradingOverlayStore.test.ts +0 -323
- package/src/trading/__tests__/UnmanagedIngestion.test.ts +0 -205
- package/src/trading/managed/ManagedTradingController.ts +0 -292
- package/src/trading/managed/managedCapabilities.ts +0 -98
- package/src/trading/managed/managedTypes.ts +0 -151
- package/src/trading/tradingTypes.ts +0 -135
- package/src/tscript/TScriptIndicator.ts +0 -54
- package/src/tscript/ast.ts +0 -105
- package/src/tscript/lexer.ts +0 -190
- package/src/tscript/parser.ts +0 -334
- package/src/tscript/runtime.ts +0 -525
- package/src/tscript/series.ts +0 -84
- package/src/types/IChart.ts +0 -56
- package/src/types/ISeries.ts +0 -30
- package/tsconfig.json +0 -23
- package/tsup.config.ts +0 -16
- package/vitest.config.ts +0 -25
|
@@ -1,188 +0,0 @@
|
|
|
1
|
-
import type { Drawing, OHLCV } from '@forgecharts/types';
|
|
2
|
-
import type { CoordTransform } from '../core/CoordTransform';
|
|
3
|
-
import { HANDLE_HALF, HANDLE_FILL, HANDLE_STROKE } from './drawingUtils';
|
|
4
|
-
|
|
5
|
-
let _counter = 0;
|
|
6
|
-
function nextId(): string {
|
|
7
|
-
return `drawing_${(++_counter).toString().padStart(4, '0')}`;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
// ─── Lines menu ───────────────────────────────────────────────────────────────
|
|
11
|
-
import { renderTrendline } from './drawing tools/lines menu/trendline';
|
|
12
|
-
import { renderRay } from './drawing tools/lines menu/ray';
|
|
13
|
-
import { renderInfoLine } from './drawing tools/lines menu/infoLine';
|
|
14
|
-
import { renderExtendedLine } from './drawing tools/lines menu/extendedLine';
|
|
15
|
-
import { renderTrendAngle } from './drawing tools/lines menu/trendAngle';
|
|
16
|
-
import { renderHorizontal } from './drawing tools/lines menu/horizontal';
|
|
17
|
-
import { renderVertical } from './drawing tools/lines menu/vertical';
|
|
18
|
-
import { renderHorizontalRay } from './drawing tools/lines menu/horizontalRay';
|
|
19
|
-
import { renderCrossLine } from './drawing tools/lines menu/crossLine';
|
|
20
|
-
// Channels
|
|
21
|
-
import { renderParallelChannel } from './drawing tools/lines menu/parallelChannel';
|
|
22
|
-
import { renderRegressionTrend } from './drawing tools/lines menu/regressionTrend';
|
|
23
|
-
import { renderFlatTopBottom } from './drawing tools/lines menu/flatTopBottom';
|
|
24
|
-
import { renderDisjointChannel } from './drawing tools/lines menu/disjointChannel';
|
|
25
|
-
// Pitchforks
|
|
26
|
-
import { renderPitchfork } from './drawing tools/lines menu/pitchfork';
|
|
27
|
-
import { renderSchiffPitchfork } from './drawing tools/lines menu/schiffPitchfork';
|
|
28
|
-
import { renderModifiedSchiffPitchfork } from './drawing tools/lines menu/modifiedSchiffPitchfork';
|
|
29
|
-
import { renderInsidePitchfork } from './drawing tools/lines menu/insidePitchfork';
|
|
30
|
-
|
|
31
|
-
// ─── Shapes menu ─────────────────────────────────────────────────────────────
|
|
32
|
-
import { renderRectangle } from './drawing tools/shapes menu/rectangle';
|
|
33
|
-
import { renderText } from './drawing tools/shapes menu/text';
|
|
34
|
-
|
|
35
|
-
// ─── Fib & Gann menu ─────────────────────────────────────────────────────────
|
|
36
|
-
import { renderFibRetracement } from './drawing tools/fib gann menu/fibRetracement';
|
|
37
|
-
|
|
38
|
-
// ─── Internal dispatch ────────────────────────────────────────────────────────
|
|
39
|
-
|
|
40
|
-
function _renderOne(ctx: CanvasRenderingContext2D, d: Drawing, t: CoordTransform, interval = '1h', candles: readonly OHLCV[] = []): void {
|
|
41
|
-
switch (d.type) {
|
|
42
|
-
// ── Lines menu ────────────────────────────────────────────────────────
|
|
43
|
-
case 'trendline': renderTrendline(ctx, d, t); break;
|
|
44
|
-
case 'ray': renderRay(ctx, d, t); break;
|
|
45
|
-
case 'infoLine': renderInfoLine(ctx, d, t, interval); break;
|
|
46
|
-
case 'extendedLine': renderExtendedLine(ctx, d, t); break;
|
|
47
|
-
case 'trendAngle': renderTrendAngle(ctx, d, t); break;
|
|
48
|
-
case 'horizontal': renderHorizontal(ctx, d, t); break;
|
|
49
|
-
case 'vertical': renderVertical(ctx, d, t); break;
|
|
50
|
-
case 'horizontalRay': renderHorizontalRay(ctx, d, t); break;
|
|
51
|
-
case 'crossLine': renderCrossLine(ctx, d, t); break;
|
|
52
|
-
// Channels
|
|
53
|
-
case 'parallelChannel': renderParallelChannel(ctx, d, t); break;
|
|
54
|
-
case 'regressionTrend': renderRegressionTrend(ctx, d, t, candles); break;
|
|
55
|
-
case 'flatTopBottom': renderFlatTopBottom(ctx, d, t); break;
|
|
56
|
-
case 'disjointChannel': renderDisjointChannel(ctx, d, t); break;
|
|
57
|
-
// Pitchforks
|
|
58
|
-
case 'pitchfork': renderPitchfork(ctx, d, t); break;
|
|
59
|
-
case 'schiffPitchfork': renderSchiffPitchfork(ctx, d, t); break;
|
|
60
|
-
case 'modifiedSchiffPitchfork': renderModifiedSchiffPitchfork(ctx, d, t); break;
|
|
61
|
-
case 'insidePitchfork': renderInsidePitchfork(ctx, d, t); break;
|
|
62
|
-
// ── Shapes menu ───────────────────────────────────────────────────────
|
|
63
|
-
case 'rectangle': renderRectangle(ctx, d, t); break;
|
|
64
|
-
case 'text': renderText(ctx, d, t); break;
|
|
65
|
-
// ── Fib & Gann menu ───────────────────────────────────────────────────
|
|
66
|
-
case 'fibRetracement': renderFibRetracement(ctx, d, t); break;
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
function _renderHandles(ctx: CanvasRenderingContext2D, d: Drawing, t: CoordTransform): void {
|
|
71
|
-
ctx.save();
|
|
72
|
-
ctx.setLineDash([]);
|
|
73
|
-
ctx.lineWidth = 1.5;
|
|
74
|
-
for (const p of d.points) {
|
|
75
|
-
const px = t.timeToX(p.time);
|
|
76
|
-
const py = t.priceToY(p.price);
|
|
77
|
-
ctx.fillStyle = HANDLE_FILL;
|
|
78
|
-
ctx.strokeStyle = HANDLE_STROKE;
|
|
79
|
-
ctx.beginPath();
|
|
80
|
-
ctx.rect(px - HANDLE_HALF, py - HANDLE_HALF, HANDLE_HALF * 2, HANDLE_HALF * 2);
|
|
81
|
-
ctx.fill();
|
|
82
|
-
ctx.stroke();
|
|
83
|
-
}
|
|
84
|
-
ctx.restore();
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
// ─── Public API ────────────────────────────────────────────────────────────────
|
|
88
|
-
|
|
89
|
-
export class DrawingManager {
|
|
90
|
-
private readonly _map = new Map<string, Drawing>();
|
|
91
|
-
|
|
92
|
-
// ─── CRUD ──────────────────────────────────────────────────────────────────
|
|
93
|
-
|
|
94
|
-
add(drawing: Omit<Drawing, 'id'>): string {
|
|
95
|
-
const id = nextId();
|
|
96
|
-
this._map.set(id, { ...drawing, id });
|
|
97
|
-
return id;
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
remove(id: string): boolean {
|
|
101
|
-
return this._map.delete(id);
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
update(
|
|
105
|
-
id: string,
|
|
106
|
-
patches: Partial<Pick<Drawing, 'points' | 'color' | 'text' | 'visible' | 'lineWidth' | 'fillColor' | 'fillOpacity' | 'deviationMultiplier'>>,
|
|
107
|
-
): boolean {
|
|
108
|
-
const d = this._map.get(id);
|
|
109
|
-
if (!d) return false;
|
|
110
|
-
this._map.set(id, { ...d, ...patches });
|
|
111
|
-
return true;
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
get(id: string): Drawing | undefined {
|
|
115
|
-
return this._map.get(id);
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
has(id: string): boolean {
|
|
119
|
-
return this._map.has(id);
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
all(): readonly Drawing[] {
|
|
123
|
-
return [...this._map.values()];
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
clear(): void {
|
|
127
|
-
this._map.clear();
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
// ─── Rendering ─────────────────────────────────────────────────────────────
|
|
131
|
-
|
|
132
|
-
render(
|
|
133
|
-
ctx: CanvasRenderingContext2D,
|
|
134
|
-
transform: CoordTransform,
|
|
135
|
-
selectedId: string | null = null,
|
|
136
|
-
interval = '1h',
|
|
137
|
-
candles: readonly OHLCV[] = [],
|
|
138
|
-
): void {
|
|
139
|
-
ctx.save();
|
|
140
|
-
for (const d of this._map.values()) {
|
|
141
|
-
if (d.visible === false) continue;
|
|
142
|
-
ctx.strokeStyle = d.color ?? '#2196f3';
|
|
143
|
-
ctx.lineWidth = d.lineWidth ?? 1.5;
|
|
144
|
-
ctx.setLineDash([]);
|
|
145
|
-
ctx.globalAlpha = 1;
|
|
146
|
-
_renderOne(ctx, d, transform, interval, candles);
|
|
147
|
-
if (d.id === selectedId) {
|
|
148
|
-
// Heavier highlight pass
|
|
149
|
-
ctx.save();
|
|
150
|
-
ctx.strokeStyle = d.color ?? '#2196f3';
|
|
151
|
-
ctx.lineWidth = (d.lineWidth ?? 1.5) + 1;
|
|
152
|
-
ctx.globalAlpha = 0.4;
|
|
153
|
-
_renderOne(ctx, d, transform, interval, candles);
|
|
154
|
-
ctx.restore();
|
|
155
|
-
_renderHandles(ctx, d, transform);
|
|
156
|
-
}
|
|
157
|
-
}
|
|
158
|
-
ctx.restore();
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
renderPreview(
|
|
162
|
-
ctx: CanvasRenderingContext2D,
|
|
163
|
-
transform: CoordTransform,
|
|
164
|
-
draft: Omit<Drawing, 'id'>,
|
|
165
|
-
interval = '1h',
|
|
166
|
-
candles: readonly OHLCV[] = [],
|
|
167
|
-
): void {
|
|
168
|
-
const d = { id: '__preview__', ...draft } as Drawing;
|
|
169
|
-
ctx.save();
|
|
170
|
-
ctx.globalAlpha = 0.65;
|
|
171
|
-
ctx.setLineDash([6, 4]);
|
|
172
|
-
ctx.strokeStyle = d.color ?? '#2196f3';
|
|
173
|
-
ctx.lineWidth = d.lineWidth ?? 1.5;
|
|
174
|
-
_renderOne(ctx, d, transform, interval, candles);
|
|
175
|
-
ctx.restore();
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
/** Returns anchor pixel positions — used by InteractionManager for handle hit-testing. */
|
|
179
|
-
static anchorPixels(
|
|
180
|
-
d: Drawing,
|
|
181
|
-
transform: CoordTransform,
|
|
182
|
-
): Array<{ x: number; y: number }> {
|
|
183
|
-
return d.points.map((p) => ({
|
|
184
|
-
x: transform.timeToX(p.time),
|
|
185
|
-
y: transform.priceToY(p.price),
|
|
186
|
-
}));
|
|
187
|
-
}
|
|
188
|
-
}
|
package/src/api/EventBus.ts
DELETED
|
@@ -1,53 +0,0 @@
|
|
|
1
|
-
import type { ChartEventMap } from '@forgecharts/types';
|
|
2
|
-
|
|
3
|
-
type Handler<T> = (payload: T) => void;
|
|
4
|
-
|
|
5
|
-
/**
|
|
6
|
-
* EventBus — internal typed pub/sub used by TChart.
|
|
7
|
-
*
|
|
8
|
-
* All event keys are constrained to `keyof ChartEventMap`, giving
|
|
9
|
-
* full type inference on payload shapes at every call site.
|
|
10
|
-
*/
|
|
11
|
-
export class EventBus {
|
|
12
|
-
private readonly _listeners = new Map<
|
|
13
|
-
keyof ChartEventMap,
|
|
14
|
-
Set<Handler<unknown>>
|
|
15
|
-
>();
|
|
16
|
-
|
|
17
|
-
on<K extends keyof ChartEventMap>(event: K, handler: Handler<ChartEventMap[K]>): void {
|
|
18
|
-
if (!this._listeners.has(event)) {
|
|
19
|
-
this._listeners.set(event, new Set());
|
|
20
|
-
}
|
|
21
|
-
this._listeners.get(event)!.add(handler as Handler<unknown>);
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
off<K extends keyof ChartEventMap>(event: K, handler: Handler<ChartEventMap[K]>): void {
|
|
25
|
-
this._listeners.get(event)?.delete(handler as Handler<unknown>);
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
once<K extends keyof ChartEventMap>(event: K, handler: Handler<ChartEventMap[K]>): void {
|
|
29
|
-
const wrapper = (payload: ChartEventMap[K]): void => {
|
|
30
|
-
handler(payload);
|
|
31
|
-
this.off(event, wrapper);
|
|
32
|
-
};
|
|
33
|
-
this.on(event, wrapper);
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
emit<K extends keyof ChartEventMap>(event: K, payload: ChartEventMap[K]): void {
|
|
37
|
-
const handlers = this._listeners.get(event);
|
|
38
|
-
if (!handlers) return;
|
|
39
|
-
for (const h of handlers) {
|
|
40
|
-
try {
|
|
41
|
-
(h as Handler<ChartEventMap[K]>)(payload);
|
|
42
|
-
} catch (err) {
|
|
43
|
-
// Prevent one bad handler from breaking the others
|
|
44
|
-
console.error(`[ForgeCharts] Unhandled error in "${event}" listener`, err);
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
/** Removes all listeners — called on chart destruction. */
|
|
50
|
-
clear(): void {
|
|
51
|
-
this._listeners.clear();
|
|
52
|
-
}
|
|
53
|
-
}
|
package/src/api/IndicatorDAG.ts
DELETED
|
@@ -1,389 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* IndicatorDAG — Directed Acyclic Graph for indicator computation.
|
|
3
|
-
*
|
|
4
|
-
* Allows indicators to be chained: e.g. EMA can consume SMA's output instead
|
|
5
|
-
* of raw OHLCV prices. Circular dependencies are rejected at registration time.
|
|
6
|
-
* Results are cached and recomputed only when inputs actually change.
|
|
7
|
-
*
|
|
8
|
-
* Incremental fast-path: when a single bar is appended, SMA and EMA extend
|
|
9
|
-
* their result by exactly one point in O(1) time.
|
|
10
|
-
*
|
|
11
|
-
* @example
|
|
12
|
-
* ```ts
|
|
13
|
-
* const dag = new IndicatorDAG();
|
|
14
|
-
* dag.addNode('sma1', { type: 'sma', params: { period: 20 } });
|
|
15
|
-
* dag.addNode('ema1', { type: 'ema', params: { period: 10 }, sourceId: 'sma1' });
|
|
16
|
-
*
|
|
17
|
-
* dag.execute(bars); // SMA first, then EMA(SMA)
|
|
18
|
-
* const result = dag.getResult('ema1'); // { kind: 'series', points: [...] }
|
|
19
|
-
* ```
|
|
20
|
-
*/
|
|
21
|
-
|
|
22
|
-
import type { OHLCV, IndicatorConfig, PriceField } from '@forgecharts/types';
|
|
23
|
-
import type { IndicatorPoint, MACDPoint } from '../core/IndicatorEngine';
|
|
24
|
-
import {
|
|
25
|
-
extractField,
|
|
26
|
-
computeSMAFromSeries,
|
|
27
|
-
computeEMAFromSeries,
|
|
28
|
-
computeWMAFromSeries,
|
|
29
|
-
computeRSI,
|
|
30
|
-
computeMACD,
|
|
31
|
-
computeVolume,
|
|
32
|
-
} from '../core/IndicatorEngine';
|
|
33
|
-
import { TScriptIndicator } from '../tscript/TScriptIndicator';
|
|
34
|
-
|
|
35
|
-
// ─── Public result type ────────────────────────────────────────────────────────
|
|
36
|
-
|
|
37
|
-
/** Computed output of a single DAG node. */
|
|
38
|
-
export type DAGResult =
|
|
39
|
-
| { readonly kind: 'series'; readonly points: readonly IndicatorPoint[] }
|
|
40
|
-
| { readonly kind: 'macd'; readonly points: readonly MACDPoint[] };
|
|
41
|
-
|
|
42
|
-
// ─── Internal ─────────────────────────────────────────────────────────────────
|
|
43
|
-
|
|
44
|
-
type Node = {
|
|
45
|
-
id: string;
|
|
46
|
-
config: IndicatorConfig;
|
|
47
|
-
result: DAGResult | null;
|
|
48
|
-
/** `bars.length` at the last successful execution. */
|
|
49
|
-
lastBarsLen: number;
|
|
50
|
-
/** Source `IndicatorPoint[]` length at the last successful execution. */
|
|
51
|
-
lastInputLen: number;
|
|
52
|
-
/** Close price of the last bar at the last successful execution — detects forming-bar live ticks. */
|
|
53
|
-
lastBarClose: number;
|
|
54
|
-
dirty: boolean;
|
|
55
|
-
/** Compiled TScript instance (only for type === 'script'). */
|
|
56
|
-
scriptRuntime?: TScriptIndicator;
|
|
57
|
-
};
|
|
58
|
-
|
|
59
|
-
function _configEqual(a: IndicatorConfig, b: IndicatorConfig): boolean {
|
|
60
|
-
return (
|
|
61
|
-
a.type === b.type &&
|
|
62
|
-
a.sourceId === b.sourceId &&
|
|
63
|
-
a.sourceField === b.sourceField &&
|
|
64
|
-
JSON.stringify(a.params) === JSON.stringify(b.params)
|
|
65
|
-
);
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
// ─── IndicatorDAG ─────────────────────────────────────────────────────────────
|
|
69
|
-
|
|
70
|
-
export class IndicatorDAG {
|
|
71
|
-
private readonly _nodes = new Map<string, Node>();
|
|
72
|
-
|
|
73
|
-
// ─── Registration ───────────────────────────────────────────────────────────
|
|
74
|
-
|
|
75
|
-
/**
|
|
76
|
-
* Registers a node for `id`.
|
|
77
|
-
* If the node already exists its config is updated if changed.
|
|
78
|
-
* Throws `'[IndicatorDAG] Circular dependency detected'` if adding `id`
|
|
79
|
-
* would form a cycle.
|
|
80
|
-
*/
|
|
81
|
-
addNode(id: string, config: IndicatorConfig): void {
|
|
82
|
-
const existing = this._nodes.get(id);
|
|
83
|
-
if (existing) {
|
|
84
|
-
if (!_configEqual(existing.config, config)) {
|
|
85
|
-
existing.config = config;
|
|
86
|
-
existing.dirty = true;
|
|
87
|
-
existing.result = null;
|
|
88
|
-
this._markDependentsDirty(id);
|
|
89
|
-
}
|
|
90
|
-
return;
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
this._nodes.set(id, {
|
|
94
|
-
id,
|
|
95
|
-
config,
|
|
96
|
-
result: null,
|
|
97
|
-
lastBarsLen: -1,
|
|
98
|
-
lastInputLen: -1,
|
|
99
|
-
lastBarClose: NaN,
|
|
100
|
-
dirty: true,
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
// Validate immediately — undo and rethrow if a cycle is formed.
|
|
104
|
-
try {
|
|
105
|
-
this._topoSort();
|
|
106
|
-
} catch (e) {
|
|
107
|
-
this._nodes.delete(id);
|
|
108
|
-
throw e;
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
/** Removes `id` and marks any indicators that depended on it as dirty. */
|
|
113
|
-
removeNode(id: string): void {
|
|
114
|
-
this._nodes.delete(id);
|
|
115
|
-
this._markDependentsDirty(id);
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
/**
|
|
119
|
-
* Syncs the DAG to exactly the provided indicator list.
|
|
120
|
-
* Adds new nodes, removes obsolete ones, updates changed configs.
|
|
121
|
-
* Throws if the resulting graph contains a cycle.
|
|
122
|
-
*/
|
|
123
|
-
sync(
|
|
124
|
-
indicators: ReadonlyArray<{ readonly id: string; readonly config: IndicatorConfig }>,
|
|
125
|
-
): void {
|
|
126
|
-
const incoming = new Set(indicators.map((i) => i.id));
|
|
127
|
-
|
|
128
|
-
// Remove nodes no longer present.
|
|
129
|
-
for (const id of [...this._nodes.keys()]) {
|
|
130
|
-
if (!incoming.has(id)) {
|
|
131
|
-
this._nodes.delete(id);
|
|
132
|
-
this._markDependentsDirty(id);
|
|
133
|
-
}
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
// Add or refresh nodes.
|
|
137
|
-
for (const { id, config } of indicators) {
|
|
138
|
-
const node = this._nodes.get(id);
|
|
139
|
-
if (!node) {
|
|
140
|
-
this._nodes.set(id, {
|
|
141
|
-
id,
|
|
142
|
-
config,
|
|
143
|
-
result: null,
|
|
144
|
-
lastBarsLen: -1,
|
|
145
|
-
lastInputLen: -1,
|
|
146
|
-
lastBarClose: NaN,
|
|
147
|
-
dirty: true,
|
|
148
|
-
});
|
|
149
|
-
} else if (!_configEqual(node.config, config)) {
|
|
150
|
-
node.config = config;
|
|
151
|
-
node.dirty = true;
|
|
152
|
-
node.result = null;
|
|
153
|
-
this._markDependentsDirty(id);
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
// Validate the whole graph (after all additions) — throws on cycle.
|
|
158
|
-
this._topoSort();
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
// ─── Execution ──────────────────────────────────────────────────────────────
|
|
162
|
-
|
|
163
|
-
/**
|
|
164
|
-
* Runs all dirty nodes in topological order.
|
|
165
|
-
*
|
|
166
|
-
* When the OHLCV array grew by exactly one bar since a node's last run and
|
|
167
|
-
* its input series also grew by one, SMA and EMA use an O(1) incremental
|
|
168
|
-
* append path. All other types fall back to a full recompute.
|
|
169
|
-
*
|
|
170
|
-
* @returns `true` if any node was executed.
|
|
171
|
-
*/
|
|
172
|
-
execute(bars: readonly OHLCV[]): boolean {
|
|
173
|
-
const order = this._topoSort();
|
|
174
|
-
let executed = false;
|
|
175
|
-
|
|
176
|
-
for (const id of order) {
|
|
177
|
-
const node = this._nodes.get(id)!;
|
|
178
|
-
if (!this._needsRun(node, bars)) continue;
|
|
179
|
-
|
|
180
|
-
executed = true;
|
|
181
|
-
const input = this._resolveSource(node, bars);
|
|
182
|
-
|
|
183
|
-
const canIncremental =
|
|
184
|
-
node.result !== null &&
|
|
185
|
-
node.result.kind === 'series' &&
|
|
186
|
-
node.result.points.length > 0 &&
|
|
187
|
-
input.length === node.lastInputLen + 1 &&
|
|
188
|
-
bars.length === node.lastBarsLen + 1;
|
|
189
|
-
|
|
190
|
-
node.result = canIncremental
|
|
191
|
-
? this._appendOne(node, input, node.result!.points as readonly IndicatorPoint[], bars)
|
|
192
|
-
: this._computeFull(node, input, bars);
|
|
193
|
-
|
|
194
|
-
node.dirty = false;
|
|
195
|
-
node.lastBarsLen = bars.length;
|
|
196
|
-
node.lastInputLen = input.length;
|
|
197
|
-
node.lastBarClose = bars.length > 0 ? bars[bars.length - 1]!.close : NaN;
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
return executed;
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
/** Forces every node to recompute on the next `execute()` call. */
|
|
204
|
-
invalidateAll(): void {
|
|
205
|
-
for (const node of this._nodes.values()) {
|
|
206
|
-
node.dirty = true;
|
|
207
|
-
}
|
|
208
|
-
}
|
|
209
|
-
|
|
210
|
-
isDirty(): boolean {
|
|
211
|
-
return [...this._nodes.values()].some((n) => n.dirty);
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
// ─── Results ────────────────────────────────────────────────────────────────
|
|
215
|
-
|
|
216
|
-
getResult(id: string): DAGResult | null {
|
|
217
|
-
return this._nodes.get(id)?.result ?? null;
|
|
218
|
-
}
|
|
219
|
-
|
|
220
|
-
getAllResults(): ReadonlyMap<string, DAGResult> {
|
|
221
|
-
const out = new Map<string, DAGResult>();
|
|
222
|
-
for (const [id, node] of this._nodes) {
|
|
223
|
-
if (node.result !== null) out.set(id, node.result);
|
|
224
|
-
}
|
|
225
|
-
return out;
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
// ─── Topological sort (Kahn's algorithm) ────────────────────────────────────
|
|
229
|
-
|
|
230
|
-
/**
|
|
231
|
-
* Returns node ids in topological execution order.
|
|
232
|
-
* Throws `'[IndicatorDAG] Circular dependency detected'` when a cycle exists.
|
|
233
|
-
*/
|
|
234
|
-
private _topoSort(): string[] {
|
|
235
|
-
const inDeg = new Map<string, number>();
|
|
236
|
-
const adj = new Map<string, string[]>(); // sourceId → dependent ids
|
|
237
|
-
|
|
238
|
-
for (const node of this._nodes.values()) {
|
|
239
|
-
if (!inDeg.has(node.id)) inDeg.set(node.id, 0);
|
|
240
|
-
const src = node.config.sourceId;
|
|
241
|
-
if (src !== undefined && this._nodes.has(src)) {
|
|
242
|
-
if (!adj.has(src)) adj.set(src, []);
|
|
243
|
-
adj.get(src)!.push(node.id);
|
|
244
|
-
inDeg.set(node.id, (inDeg.get(node.id) ?? 0) + 1);
|
|
245
|
-
}
|
|
246
|
-
}
|
|
247
|
-
|
|
248
|
-
const queue = [...this._nodes.keys()].filter((id) => (inDeg.get(id) ?? 0) === 0);
|
|
249
|
-
const order: string[] = [];
|
|
250
|
-
|
|
251
|
-
while (queue.length > 0) {
|
|
252
|
-
const curr = queue.shift()!;
|
|
253
|
-
order.push(curr);
|
|
254
|
-
for (const dep of adj.get(curr) ?? []) {
|
|
255
|
-
const d = (inDeg.get(dep) ?? 0) - 1;
|
|
256
|
-
inDeg.set(dep, d);
|
|
257
|
-
if (d === 0) queue.push(dep);
|
|
258
|
-
}
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
if (order.length !== this._nodes.size) {
|
|
262
|
-
throw new Error('[IndicatorDAG] Circular dependency detected');
|
|
263
|
-
}
|
|
264
|
-
return order;
|
|
265
|
-
}
|
|
266
|
-
|
|
267
|
-
// ─── Private helpers ────────────────────────────────────────────────────────
|
|
268
|
-
|
|
269
|
-
private _needsRun(node: Node, bars: readonly OHLCV[]): boolean {
|
|
270
|
-
if (node.dirty || node.result === null) return true;
|
|
271
|
-
if (bars.length !== node.lastBarsLen) return true;
|
|
272
|
-
// Rerun when the forming bar's close has changed (live WS tick updating in-place).
|
|
273
|
-
if (bars.length > 0 && bars[bars.length - 1]!.close !== node.lastBarClose) return true;
|
|
274
|
-
// Rerun if the source indicator has newer data.
|
|
275
|
-
const srcId = node.config.sourceId;
|
|
276
|
-
if (srcId !== undefined) {
|
|
277
|
-
const src = this._nodes.get(srcId);
|
|
278
|
-
if (src && src.lastBarsLen !== node.lastBarsLen) return true;
|
|
279
|
-
}
|
|
280
|
-
return false;
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
/** Returns the ordered input series for `node` — either from a source indicator or raw price field. */
|
|
284
|
-
private _resolveSource(node: Node, bars: readonly OHLCV[]): readonly IndicatorPoint[] {
|
|
285
|
-
const srcId = node.config.sourceId;
|
|
286
|
-
if (srcId !== undefined) {
|
|
287
|
-
const src = this._nodes.get(srcId);
|
|
288
|
-
if (src?.result?.kind === 'series') return src.result.points;
|
|
289
|
-
// Source is missing, MACD, or not yet computed — fall through to raw bars.
|
|
290
|
-
}
|
|
291
|
-
const field: PriceField = node.config.sourceField ?? 'close';
|
|
292
|
-
return bars.map((b) => ({ time: b.time, value: extractField(b, field) }));
|
|
293
|
-
}
|
|
294
|
-
|
|
295
|
-
/** Full recompute of a node's result. */
|
|
296
|
-
private _computeFull(
|
|
297
|
-
node: Node,
|
|
298
|
-
input: readonly IndicatorPoint[],
|
|
299
|
-
bars: readonly OHLCV[],
|
|
300
|
-
): DAGResult {
|
|
301
|
-
const p = (node.config.params ?? {}) as Record<string, number>;
|
|
302
|
-
switch (node.config.type) {
|
|
303
|
-
case 'sma':
|
|
304
|
-
return { kind: 'series', points: computeSMAFromSeries(input, p['period'] ?? 20) };
|
|
305
|
-
case 'ema':
|
|
306
|
-
return { kind: 'series', points: computeEMAFromSeries(input, p['period'] ?? 20) };
|
|
307
|
-
case 'wma':
|
|
308
|
-
return { kind: 'series', points: computeWMAFromSeries(input, p['period'] ?? 20) };
|
|
309
|
-
case 'rsi':
|
|
310
|
-
return { kind: 'series', points: computeRSI(bars, p['period'] ?? 14) };
|
|
311
|
-
case 'volume':
|
|
312
|
-
return { kind: 'series', points: computeVolume(bars) };
|
|
313
|
-
case 'macd':
|
|
314
|
-
return {
|
|
315
|
-
kind: 'macd',
|
|
316
|
-
points: computeMACD(bars, p['fast'] ?? 12, p['slow'] ?? 26, p['signal'] ?? 9),
|
|
317
|
-
};
|
|
318
|
-
case 'bbands': {
|
|
319
|
-
// Bollinger Bands — represented as three series; store middle band as primary.
|
|
320
|
-
const period = p['period'] ?? 20;
|
|
321
|
-
const sma = computeSMAFromSeries(input, period);
|
|
322
|
-
return { kind: 'series', points: sma }; // renderer handles bbands specially
|
|
323
|
-
}
|
|
324
|
-
case 'script': {
|
|
325
|
-
const src = node.config.script;
|
|
326
|
-
if (!src) return { kind: 'series', points: [] };
|
|
327
|
-
// Compile once, reuse across bars.
|
|
328
|
-
if (!node.scriptRuntime) {
|
|
329
|
-
node.scriptRuntime = new TScriptIndicator(src);
|
|
330
|
-
}
|
|
331
|
-
const plots = node.scriptRuntime.run(bars);
|
|
332
|
-
// First plot() call becomes the primary series.
|
|
333
|
-
return { kind: 'series', points: plots[0] ?? [] };
|
|
334
|
-
}
|
|
335
|
-
default:
|
|
336
|
-
// Identity passthrough for unknown / future types.
|
|
337
|
-
return { kind: 'series', points: [...input] };
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
|
|
341
|
-
/**
|
|
342
|
-
* O(1) append of a single new point.
|
|
343
|
-
* Only called when `canIncremental` is true (input grew by exactly one element).
|
|
344
|
-
* Falls back to `_computeFull` for types that have no O(1) path.
|
|
345
|
-
*/
|
|
346
|
-
private _appendOne(
|
|
347
|
-
node: Node,
|
|
348
|
-
input: readonly IndicatorPoint[],
|
|
349
|
-
prev: readonly IndicatorPoint[],
|
|
350
|
-
bars: readonly OHLCV[],
|
|
351
|
-
): DAGResult {
|
|
352
|
-
const p = (node.config.params ?? {}) as Record<string, number>;
|
|
353
|
-
const last = input[input.length - 1]!;
|
|
354
|
-
|
|
355
|
-
switch (node.config.type) {
|
|
356
|
-
case 'sma': {
|
|
357
|
-
const period = p['period'] ?? 20;
|
|
358
|
-
if (input.length < period || prev.length === 0) {
|
|
359
|
-
return this._computeFull(node, input, bars);
|
|
360
|
-
}
|
|
361
|
-
// O(1): newSMA = (prevSMA * period - outgoing + incoming) / period
|
|
362
|
-
const prevSMA = prev[prev.length - 1]!.value;
|
|
363
|
-
const outgoing = input[input.length - 1 - period]?.value ?? 0;
|
|
364
|
-
const newValue = (prevSMA * period - outgoing + last.value) / period;
|
|
365
|
-
return { kind: 'series', points: [...prev, { time: last.time, value: newValue }] };
|
|
366
|
-
}
|
|
367
|
-
case 'ema': {
|
|
368
|
-
const period = p['period'] ?? 20;
|
|
369
|
-
const k = 2 / (period + 1);
|
|
370
|
-
const prevEMA = prev[prev.length - 1]!.value;
|
|
371
|
-
const newValue = last.value * k + prevEMA * (1 - k);
|
|
372
|
-
return { kind: 'series', points: [...prev, { time: last.time, value: newValue }] };
|
|
373
|
-
}
|
|
374
|
-
default:
|
|
375
|
-
return this._computeFull(node, input, bars);
|
|
376
|
-
}
|
|
377
|
-
}
|
|
378
|
-
|
|
379
|
-
/** Recursively marks all indicators that depend on `sourceId` as dirty. */
|
|
380
|
-
private _markDependentsDirty(sourceId: string): void {
|
|
381
|
-
for (const node of this._nodes.values()) {
|
|
382
|
-
if (node.config.sourceId === sourceId && !node.dirty) {
|
|
383
|
-
node.dirty = true;
|
|
384
|
-
node.result = null;
|
|
385
|
-
this._markDependentsDirty(node.id);
|
|
386
|
-
}
|
|
387
|
-
}
|
|
388
|
-
}
|
|
389
|
-
}
|
|
@@ -1,47 +0,0 @@
|
|
|
1
|
-
import type { IndicatorConfig, IndicatorInstance } from '@forgecharts/types';
|
|
2
|
-
|
|
3
|
-
let _counter = 0;
|
|
4
|
-
function nextId(): string {
|
|
5
|
-
return `indicator_${(++_counter).toString().padStart(4, '0')}`;
|
|
6
|
-
}
|
|
7
|
-
|
|
8
|
-
/**
|
|
9
|
-
* IndicatorRegistry — tracks active indicator instances for a chart.
|
|
10
|
-
* Rendering is a separate concern (handled by sub-pane renderers, out of SDK scope).
|
|
11
|
-
*/
|
|
12
|
-
export class IndicatorRegistry {
|
|
13
|
-
private readonly _map = new Map<string, IndicatorInstance>();
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
* Registers a new indicator and returns its opaque id.
|
|
17
|
-
*/
|
|
18
|
-
add(config: IndicatorConfig): string {
|
|
19
|
-
const id = nextId();
|
|
20
|
-
this._map.set(id, { id, config });
|
|
21
|
-
return id;
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
/**
|
|
25
|
-
* Removes an indicator by id.
|
|
26
|
-
* @returns `true` if the indicator existed and was removed.
|
|
27
|
-
*/
|
|
28
|
-
remove(id: string): boolean {
|
|
29
|
-
return this._map.delete(id);
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
get(id: string): IndicatorInstance | undefined {
|
|
33
|
-
return this._map.get(id);
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
has(id: string): boolean {
|
|
37
|
-
return this._map.has(id);
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
all(): readonly IndicatorInstance[] {
|
|
41
|
-
return [...this._map.values()];
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
clear(): void {
|
|
45
|
-
this._map.clear();
|
|
46
|
-
}
|
|
47
|
-
}
|