2023-12-13 01:33:41 +00:00
|
|
|
import { ChartType, Overlay, Study } from 'pennant';
|
2023-08-02 14:29:41 +00:00
|
|
|
import { create } from 'zustand';
|
|
|
|
import { persist } from 'zustand/middleware';
|
|
|
|
import { immer } from 'zustand/middleware/immer';
|
2023-12-13 01:33:41 +00:00
|
|
|
import { Interval } from '@vegaprotocol/types';
|
|
|
|
import { getValidItem, getValidSubset } from '@vegaprotocol/react-helpers';
|
2023-08-02 14:29:41 +00:00
|
|
|
|
2023-09-12 16:56:43 +00:00
|
|
|
type StudySizes = { [S in Study]?: number };
|
2023-12-13 01:33:41 +00:00
|
|
|
export type Chartlib = 'pennant' | 'tradingview';
|
2023-09-12 16:56:43 +00:00
|
|
|
|
2023-08-02 14:29:41 +00:00
|
|
|
interface StoredSettings {
|
2024-01-24 10:21:33 +00:00
|
|
|
state: object | undefined; // Don't see a better type provided from TradingView type definitions
|
2023-12-13 01:33:41 +00:00
|
|
|
chartlib: Chartlib;
|
|
|
|
// For interval we use the enum from @vegaprotocol/types, this is to make mapping between different
|
|
|
|
// chart types easier and more consistent
|
2023-08-02 14:29:41 +00:00
|
|
|
interval: Interval;
|
|
|
|
type: ChartType;
|
|
|
|
overlays: Overlay[];
|
|
|
|
studies: Study[];
|
2023-09-12 16:56:43 +00:00
|
|
|
studySizes: StudySizes;
|
2023-08-02 14:29:41 +00:00
|
|
|
}
|
|
|
|
|
2023-09-15 12:24:47 +00:00
|
|
|
export const STUDY_SIZE = 90;
|
2023-09-12 16:56:43 +00:00
|
|
|
const STUDY_ORDER: Study[] = [
|
|
|
|
Study.FORCE_INDEX,
|
|
|
|
Study.RELATIVE_STRENGTH_INDEX,
|
|
|
|
Study.ELDAR_RAY,
|
|
|
|
Study.MACD,
|
|
|
|
Study.VOLUME,
|
|
|
|
];
|
|
|
|
|
2023-10-27 13:21:14 +00:00
|
|
|
export const DEFAULT_CHART_SETTINGS = {
|
2024-01-24 10:21:33 +00:00
|
|
|
state: undefined,
|
|
|
|
chartlib: 'pennant' as const,
|
2023-12-13 01:33:41 +00:00
|
|
|
interval: Interval.INTERVAL_I15M,
|
2023-08-02 14:29:41 +00:00
|
|
|
type: ChartType.CANDLE,
|
2023-09-05 00:21:46 +00:00
|
|
|
overlays: [Overlay.MOVING_AVERAGE],
|
|
|
|
studies: [Study.MACD, Study.VOLUME],
|
2023-09-12 16:56:43 +00:00
|
|
|
studySizes: {},
|
2023-08-02 14:29:41 +00:00
|
|
|
};
|
|
|
|
|
2023-12-13 01:33:41 +00:00
|
|
|
export const useChartSettingsStore = create<
|
2023-08-02 14:29:41 +00:00
|
|
|
StoredSettings & {
|
|
|
|
setType: (type: ChartType) => void;
|
|
|
|
setInterval: (interval: Interval) => void;
|
2023-09-12 16:56:43 +00:00
|
|
|
setOverlays: (overlays?: Overlay[]) => void;
|
|
|
|
setStudies: (studies?: Study[]) => void;
|
|
|
|
setStudySizes: (sizes: number[]) => void;
|
2023-12-13 01:33:41 +00:00
|
|
|
setChartlib: (lib: Chartlib) => void;
|
2024-01-24 10:21:33 +00:00
|
|
|
setState: (state: object) => void;
|
2023-08-02 14:29:41 +00:00
|
|
|
}
|
|
|
|
>()(
|
|
|
|
persist(
|
|
|
|
immer((set) => ({
|
|
|
|
...DEFAULT_CHART_SETTINGS,
|
|
|
|
setType: (type) =>
|
|
|
|
set((state) => {
|
|
|
|
state.type = type;
|
|
|
|
}),
|
|
|
|
setInterval: (interval) =>
|
|
|
|
set((state) => {
|
|
|
|
state.interval = interval;
|
|
|
|
}),
|
2023-09-12 16:56:43 +00:00
|
|
|
setOverlays: (overlays) => {
|
|
|
|
if (!overlays) return;
|
|
|
|
|
2023-08-02 14:29:41 +00:00
|
|
|
set((state) => {
|
|
|
|
state.overlays = overlays;
|
2023-09-12 16:56:43 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
setStudies: (studies) => {
|
|
|
|
if (!studies) return;
|
|
|
|
|
|
|
|
// Make sure studies are always returned in the same order
|
|
|
|
studies.sort((a, b) => {
|
|
|
|
return STUDY_ORDER.indexOf(a) - STUDY_ORDER.indexOf(b);
|
|
|
|
});
|
|
|
|
|
2023-08-02 14:29:41 +00:00
|
|
|
set((state) => {
|
|
|
|
state.studies = studies;
|
2023-09-12 16:56:43 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
setStudySizes: (sizes) => {
|
|
|
|
set((state) => {
|
|
|
|
// for every study find the corresonding size and update
|
|
|
|
// the size record for that study
|
|
|
|
state.studies.forEach((s, i) => {
|
|
|
|
const size = sizes[i];
|
|
|
|
state.studySizes[s] = size;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
2023-12-13 01:33:41 +00:00
|
|
|
setChartlib: (lib) => {
|
|
|
|
set((state) => {
|
|
|
|
state.chartlib = lib;
|
|
|
|
});
|
|
|
|
},
|
2024-01-24 10:21:33 +00:00
|
|
|
setState: (state) => {
|
|
|
|
set({ state });
|
2023-12-13 01:33:41 +00:00
|
|
|
},
|
2023-08-02 14:29:41 +00:00
|
|
|
})),
|
|
|
|
{
|
|
|
|
name: 'vega_candles_chart_store',
|
|
|
|
}
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2023-12-13 01:33:41 +00:00
|
|
|
export const useChartSettings = () => {
|
|
|
|
const settings = useChartSettingsStore();
|
2023-08-02 14:29:41 +00:00
|
|
|
|
|
|
|
const interval: Interval = getValidItem(
|
|
|
|
settings.interval,
|
|
|
|
Object.values(Interval),
|
2023-12-13 01:33:41 +00:00
|
|
|
Interval.INTERVAL_I15M
|
2023-08-02 14:29:41 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const chartType: ChartType = getValidItem(
|
|
|
|
settings.type,
|
|
|
|
Object.values(ChartType),
|
|
|
|
ChartType.CANDLE
|
|
|
|
);
|
|
|
|
|
|
|
|
const overlays: Overlay[] = getValidSubset(
|
|
|
|
settings.overlays,
|
|
|
|
Object.values(Overlay),
|
|
|
|
[]
|
|
|
|
);
|
|
|
|
|
|
|
|
const studies: Study[] = getValidSubset(
|
|
|
|
settings.studies,
|
|
|
|
Object.values(Study),
|
|
|
|
[Study.VOLUME]
|
|
|
|
);
|
|
|
|
|
2023-09-12 16:56:43 +00:00
|
|
|
// find the study size
|
|
|
|
const studySizes = studies.map((s) => {
|
|
|
|
const size = settings.studySizes[s] || STUDY_SIZE;
|
|
|
|
return size;
|
|
|
|
});
|
|
|
|
|
2023-08-02 14:29:41 +00:00
|
|
|
return {
|
2023-12-13 01:33:41 +00:00
|
|
|
chartlib: settings.chartlib,
|
2023-08-02 14:29:41 +00:00
|
|
|
interval,
|
|
|
|
chartType,
|
|
|
|
overlays,
|
|
|
|
studies,
|
2023-09-12 16:56:43 +00:00
|
|
|
studySizes,
|
2023-08-02 14:29:41 +00:00
|
|
|
setInterval: settings.setInterval,
|
|
|
|
setType: settings.setType,
|
|
|
|
setStudies: settings.setStudies,
|
|
|
|
setOverlays: settings.setOverlays,
|
2023-09-12 16:56:43 +00:00
|
|
|
setStudySizes: settings.setStudySizes,
|
2023-12-13 01:33:41 +00:00
|
|
|
setChartlib: settings.setChartlib,
|
2024-01-24 10:21:33 +00:00
|
|
|
state: settings.state,
|
|
|
|
setState: settings.setState,
|
2023-08-02 14:29:41 +00:00
|
|
|
};
|
|
|
|
};
|