在以下 Hook 中添加请求取消逻辑,防止快速切换股票时旧数据覆盖新数据: - useBasicInfo - useShareholderData - useManagementData - useBranchesData - useAnnouncementsData - useDisclosureData - useStockQuoteData 修复前:stockCode 变化时,旧请求可能后返回,覆盖新数据 修复后:cleanup 时取消旧请求,确保数据一致性 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
64 lines
1.6 KiB
TypeScript
64 lines
1.6 KiB
TypeScript
// src/views/Company/components/CompanyOverview/hooks/useBasicInfo.ts
|
|
// 公司基本信息 Hook - 用于 CompanyHeaderCard
|
|
|
|
import { useState, useEffect } from "react";
|
|
import { logger } from "@utils/logger";
|
|
import axios from "@utils/axiosConfig";
|
|
import type { BasicInfo } from "../types";
|
|
|
|
interface ApiResponse<T> {
|
|
success: boolean;
|
|
data: T;
|
|
}
|
|
|
|
interface UseBasicInfoResult {
|
|
basicInfo: BasicInfo | null;
|
|
loading: boolean;
|
|
error: string | null;
|
|
}
|
|
|
|
/**
|
|
* 公司基本信息 Hook
|
|
* @param stockCode - 股票代码
|
|
*/
|
|
export const useBasicInfo = (stockCode?: string): UseBasicInfoResult => {
|
|
const [basicInfo, setBasicInfo] = useState<BasicInfo | null>(null);
|
|
const [loading, setLoading] = useState(false);
|
|
const [error, setError] = useState<string | null>(null);
|
|
|
|
useEffect(() => {
|
|
if (!stockCode) return;
|
|
|
|
const controller = new AbortController();
|
|
|
|
const loadData = async () => {
|
|
setLoading(true);
|
|
setError(null);
|
|
|
|
try {
|
|
const { data: result } = await axios.get<ApiResponse<BasicInfo>>(
|
|
`/api/stock/${stockCode}/basic-info`,
|
|
{ signal: controller.signal }
|
|
);
|
|
|
|
if (result.success) {
|
|
setBasicInfo(result.data);
|
|
} else {
|
|
setError("加载基本信息失败");
|
|
}
|
|
} catch (err: any) {
|
|
if (err.name === "CanceledError") return;
|
|
logger.error("useBasicInfo", "loadData", err, { stockCode });
|
|
setError("网络请求失败");
|
|
} finally {
|
|
setLoading(false);
|
|
}
|
|
};
|
|
|
|
loadData();
|
|
return () => controller.abort();
|
|
}, [stockCode]);
|
|
|
|
return { basicInfo, loading, error };
|
|
};
|