ExamCard.tsx 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. import React,{ useState, useCallback, useEffect } from 'react';
  2. import { useQuery } from '@tanstack/react-query';
  3. import { useSearchParams } from "react-router";
  4. import dayjs from 'dayjs';
  5. import { message } from 'antd';
  6. import { useCurrentQuestion, useRoomMessages, useAnswerSubmission, useUserAnswerHistory, useSocketRoom } from './hooks/useSocketClient.ts';
  7. import type { AnswerRecord, ClassroomData, Answer } from './types.ts';
  8. // 答题卡页面
  9. export default function ExamCard() {
  10. const [searchParams] = useSearchParams();
  11. const classroom = searchParams.get('classroom');
  12. const nickname = searchParams.get('nickname');
  13. useSocketRoom(classroom);
  14. const lastMessage = useRoomMessages(classroom);
  15. const { submitAnswer } = useAnswerSubmission(classroom);
  16. const {currentQuestion} = useCurrentQuestion(classroom);
  17. const { userAnswers } = useUserAnswerHistory(classroom, nickname);
  18. const [currentDate, setCurrentDate] = useState('');
  19. const [currentPrice, setCurrentPrice] = useState('0');
  20. const [holdingStock, setHoldingStock] = useState('0');
  21. const [holdingCash, setHoldingCash] = useState('0');
  22. const [isStarted, setIsStarted] = useState(false);
  23. const [answerRecords, setAnswerRecords] = useState<AnswerRecord[]>([]);
  24. const { data: classroomData, isLoading } = useQuery({
  25. queryKey: ['classroom', classroom],
  26. queryFn: async () => {
  27. if (!classroom) return null;
  28. const response = await fetch(`/api/v1/classroom?classroom_no=${classroom}`);
  29. const data = await response.json();
  30. if (!data.success) {
  31. message.error(data.message || '获取教室数据失败');
  32. return null;
  33. }
  34. return data.data?.[0] as ClassroomData || null;
  35. },
  36. enabled: !!classroom
  37. });
  38. useEffect(() => {
  39. if (!classroom || !nickname) {
  40. globalThis.location.href = '/exam';
  41. return;
  42. }
  43. if (classroomData && classroomData.status !== "1") {
  44. message.error('该教室已关闭');
  45. globalThis.location.href = '/exam';
  46. return;
  47. }
  48. }, [classroom, nickname, classroomData]);
  49. // 处理房间消息
  50. useEffect(() => {
  51. if (!lastMessage?.message) return;
  52. const { type } = lastMessage.message;
  53. // 只处理重开消息的UI重置
  54. if (type === 'restart') {
  55. setCurrentDate('');
  56. setCurrentPrice('0');
  57. setHoldingStock('0');
  58. setHoldingCash('0');
  59. setIsStarted(false);
  60. setAnswerRecords([]);
  61. }
  62. }, [lastMessage]);
  63. useEffect(() => {
  64. if (currentQuestion) {
  65. console.log('currentQuestion', currentQuestion);
  66. setCurrentDate(currentQuestion.date);
  67. setCurrentPrice(String(currentQuestion.price));
  68. setIsStarted(true);
  69. } else {
  70. // 如果没有当前问题,重置状态
  71. setCurrentDate('');
  72. setCurrentPrice('0');
  73. setIsStarted(false);
  74. }
  75. }, [currentQuestion]);
  76. // 处理选择A(持股)
  77. const handleChooseA = useCallback(async () => {
  78. setHoldingStock('1');
  79. setHoldingCash('0');
  80. if (classroom && nickname) {
  81. const answer = {
  82. date: currentDate,
  83. holdingStock: '1',
  84. holdingCash: '0',
  85. userId: nickname,
  86. price: currentPrice
  87. };
  88. try {
  89. await submitAnswer(currentDate, nickname, answer);
  90. } catch (error) {
  91. message.error('提交答案失败');
  92. }
  93. }
  94. }, [classroom, nickname, currentDate, currentPrice, submitAnswer]);
  95. // 处理选择B(持币)
  96. const handleChooseB = useCallback(async () => {
  97. setHoldingStock('0');
  98. setHoldingCash('1');
  99. if (classroom && nickname) {
  100. const answer = {
  101. date: currentDate,
  102. holdingStock: '0',
  103. holdingCash: '1',
  104. userId: nickname,
  105. price: currentPrice
  106. };
  107. try {
  108. await submitAnswer(currentDate, nickname, answer);
  109. } catch (error) {
  110. message.error('提交答案失败');
  111. }
  112. }
  113. }, [classroom, nickname, currentDate, currentPrice, submitAnswer]);
  114. // 初始化用户的答题记录
  115. useEffect(() => {
  116. if (userAnswers && userAnswers.length > 0) {
  117. const lastAnswer = userAnswers[userAnswers.length - 1];
  118. setHoldingStock(lastAnswer.holdingStock);
  119. setHoldingCash(lastAnswer.holdingCash);
  120. // 直接使用 userAnswers 中已计算好的数据
  121. const records = userAnswers.map((answer: Answer, index: number): AnswerRecord => ({
  122. date: answer.date,
  123. price: String(answer.price || '0'),
  124. holdingStock: answer.holdingStock,
  125. holdingCash: answer.holdingCash,
  126. profitAmount: answer.profitAmount || 0,
  127. profitPercent: answer.profitPercent || 0,
  128. index: index + 1
  129. }));
  130. setAnswerRecords(records);
  131. }
  132. }, [userAnswers]);
  133. if (isLoading || !classroomData) {
  134. return <div className="flex items-center justify-center min-h-screen">加载中...</div>;
  135. }
  136. return (
  137. <div className="flex flex-col items-center min-h-screen bg-gray-100 py-8">
  138. {/* 选择区域 */}
  139. <div className="w-full max-w-2xl">
  140. <div className="text-center mb-8">
  141. <h2 className="text-2xl font-bold mb-2">持股选A, 持币选B</h2>
  142. <div className="flex justify-center space-x-4 text-gray-600">
  143. {isStarted ? (
  144. <>
  145. <span>日期: {currentDate}</span>
  146. <span>价格: {currentPrice}</span>
  147. </>
  148. ) : (
  149. <div className="text-blue-600">
  150. <div className="mb-2">等待训练开始...</div>
  151. <div className="text-sm text-gray-500">
  152. 训练日期: {dayjs(classroomData.training_date).format('YYYY-MM-DD')}
  153. </div>
  154. </div>
  155. )}
  156. </div>
  157. </div>
  158. {/* 选择按钮 */}
  159. <div className="flex justify-center items-center space-x-4 mb-8 bg-white p-6 rounded-lg shadow-md">
  160. <button
  161. onClick={handleChooseA}
  162. disabled={!isStarted}
  163. className={`flex-1 py-8 text-3xl font-bold rounded-lg transition-colors ${
  164. !isStarted
  165. ? 'bg-gray-200 text-gray-400 cursor-not-allowed'
  166. : holdingStock === '1'
  167. ? 'bg-red-500 text-white'
  168. : 'bg-gray-100 text-gray-700 hover:bg-gray-200'
  169. }`}
  170. >
  171. A
  172. </button>
  173. <div className="text-xl font-medium text-gray-700">
  174. {isStarted ? '开始' : '等待'}
  175. </div>
  176. <button
  177. onClick={handleChooseB}
  178. disabled={!isStarted}
  179. className={`flex-1 py-8 text-3xl font-bold rounded-lg transition-colors ${
  180. !isStarted
  181. ? 'bg-gray-200 text-gray-400 cursor-not-allowed'
  182. : holdingCash === '1'
  183. ? 'bg-green-500 text-white'
  184. : 'bg-gray-100 text-gray-700 hover:bg-gray-200'
  185. }`}
  186. >
  187. B
  188. </button>
  189. </div>
  190. {/* 信息显示 */}
  191. <div className="bg-white p-6 rounded-lg shadow-md">
  192. <div className="grid grid-cols-2 gap-4 mb-4">
  193. <div className="text-gray-600">昵称: {nickname}</div>
  194. <div className="text-gray-600">代码: {classroomData.code}</div>
  195. </div>
  196. {/* 表格头部 */}
  197. <div className="grid grid-cols-8 gap-4 py-2 border-b border-gray-200 text-sm font-medium text-gray-600">
  198. <div>序</div>
  199. <div>训练日期</div>
  200. <div>持股</div>
  201. <div>持币</div>
  202. <div>价格</div>
  203. <div>收益(元)</div>
  204. <div>盈亏率</div>
  205. <div>号</div>
  206. </div>
  207. {/* 表格内容 */}
  208. <div className="max-h-60 overflow-y-auto">
  209. {[...answerRecords].reverse().map((record: AnswerRecord) => (
  210. <div key={record.date} className="grid grid-cols-8 gap-4 py-2 text-sm text-gray-800 hover:bg-gray-50">
  211. <div>{record.index}</div>
  212. <div>{dayjs(record.date).format('YYYY-MM-DD')}</div>
  213. <div className="text-red-500">{record.holdingStock}</div>
  214. <div className="text-green-500">{record.holdingCash}</div>
  215. <div>{record.price}</div>
  216. <div className={record.profitAmount >= 0 ? 'text-red-500' : 'text-green-500'}>
  217. {record.profitAmount.toFixed(2)}
  218. </div>
  219. <div className={record.profitPercent >= 0 ? 'text-red-500' : 'text-green-500'}>
  220. {record.profitPercent.toFixed(2)}%
  221. </div>
  222. <div>{record.index}</div>
  223. </div>
  224. ))}
  225. </div>
  226. </div>
  227. </div>
  228. </div>
  229. );
  230. }