Files
auto-trade/features/dashboard/hooks/use-holdings-realtime.ts

82 lines
2.5 KiB
TypeScript
Raw Normal View History

2026-02-13 12:17:35 +09:00
"use client";
import { useEffect, useMemo, useRef, useState } from "react";
2026-02-13 12:17:35 +09:00
import type { DashboardHoldingItem } from "@/features/dashboard/types/dashboard.types";
import {
type KisRealtimeStockTick,
parseKisRealtimeStockTick,
} from "@/features/dashboard/utils/kis-stock-realtime.utils";
import { useKisWebSocketStore } from "@/features/kis-realtime/stores/kisWebSocketStore";
const STOCK_REALTIME_TR_ID = "H0STCNT0";
/**
* @description .
* @param holdings
* @returns /
* @remarks UI 흐름: DashboardContainer -> useHoldingsRealtime -> HoldingsList/summary
* @see features/dashboard/components/DashboardContainer.tsx
*/
export function useHoldingsRealtime(holdings: DashboardHoldingItem[]) {
const [realtimeData, setRealtimeData] = useState<
Record<string, KisRealtimeStockTick>
>({});
const subscribeRef = useRef(useKisWebSocketStore.getState().subscribe);
const connectRef = useRef(useKisWebSocketStore.getState().connect);
const { isConnected } = useKisWebSocketStore();
2026-02-13 12:17:35 +09:00
const uniqueSymbols = useMemo(
() =>
Array.from(new Set((holdings ?? []).map((item) => item.symbol))).sort(),
2026-02-13 12:17:35 +09:00
[holdings],
);
const symbolKey = useMemo(() => uniqueSymbols.join(","), [uniqueSymbols]);
useEffect(() => {
if (uniqueSymbols.length === 0) {
const resetTimerId = window.setTimeout(() => {
setRealtimeData({});
}, 0);
return () => window.clearTimeout(resetTimerId);
}
connectRef.current();
2026-02-13 12:17:35 +09:00
const unsubs: (() => void)[] = [];
uniqueSymbols.forEach((symbol) => {
const unsub = subscribeRef.current(
STOCK_REALTIME_TR_ID,
symbol,
(data: string) => {
const tick = parseKisRealtimeStockTick(data);
if (tick) {
setRealtimeData((prev) => {
const prevTick = prev[tick.symbol];
if (
prevTick?.currentPrice === tick.currentPrice &&
prevTick?.change === tick.change &&
prevTick?.changeRate === tick.changeRate
) {
return prev;
}
2026-02-13 12:17:35 +09:00
return {
...prev,
[tick.symbol]: tick,
};
});
}
},
);
2026-02-13 12:17:35 +09:00
unsubs.push(unsub);
});
return () => {
unsubs.forEach((unsub) => unsub());
};
}, [symbolKey, uniqueSymbols]);
2026-02-13 12:17:35 +09:00
return { realtimeData, isConnected };
}