You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

682 lines
18 KiB

4 weeks ago
4 weeks ago
  1. <!-- @format -->
  2. <template>
  3. <view class="main">
  4. <!-- 自定义导航栏 -->
  5. <view class="header_fixed" :style="{ top: iSMT + 'px' }">
  6. <view class="header-content">
  7. <view class="header-left" @click="goBack">
  8. <text class="back-text"></text>
  9. </view>
  10. <view class="header-center">
  11. <text class="header-title">{{ marketTitle }}</text>
  12. </view>
  13. <view class="header-right">
  14. <image src="/static/marketSituation-image/search.png" class="header-icon" mode="aspectFit"></image>
  15. <text class="more-text">···</text>
  16. </view>
  17. </view>
  18. <!-- 表头 -->
  19. <view class="table-header">
  20. <view class="header-item name-column">
  21. <text class="header-text">名称</text>
  22. </view>
  23. <view class="header-item price-column" @click="sortByPrice">
  24. <text class="header-text">最新</text>
  25. <text class="sort-icon">{{ sortType === "price" ? (sortOrder === "asc" ? "↑" : "↓") : "↕" }}</text>
  26. </view>
  27. <view class="header-item change-column" @click="sortByChange">
  28. <text class="header-text">涨幅</text>
  29. <text class="sort-icon">{{ sortType === "change" ? (sortOrder === "asc" ? "↑" : "↓") : "↕" }}</text>
  30. </view>
  31. </view>
  32. </view>
  33. <!-- 内容区域 -->
  34. <scroll-view class="content" :style="{ top: contentTopPosition + 'px' }" scroll-y="true">
  35. <!-- 股票列表 -->
  36. <view class="stock-list">
  37. <view class="stock-row" v-for="(item, index) in sortedStockList" :key="item" @click="viewIndexDetail(item, index)">
  38. <view class="stock-cell name-column">
  39. <view class="stock-name">{{ item.stockName }}</view>
  40. <view class="stock-code">{{ item.stockCode }}</view>
  41. </view>
  42. <view class="stock-cell price-column">
  43. <text class="stock-price" :class="item.isRising ? 'rising' : 'falling'">
  44. {{ typeof item.currentPrice === "number" ? item.currentPrice.toFixed(2) : item.currentPrice }}
  45. </text>
  46. </view>
  47. <view class="stock-cell change-column">
  48. <text class="stock-change" :class="item.isRising ? 'rising' : 'falling'">
  49. {{ judgeSymbol(item.changePercent) }}
  50. </text>
  51. </view>
  52. </view>
  53. </view>
  54. <!-- 底部安全区域 -->
  55. <!-- <view class="bottom-safe-area"></view> -->
  56. </scroll-view>
  57. </view>
  58. <!-- 底部导航栏 -->
  59. <!-- <footerBar class="static-footer" :type="'marketSituation'"></footerBar> -->
  60. </template>
  61. <script setup>
  62. import { ref, computed, onMounted, onUnmounted, nextTick, watch } from "vue";
  63. import { onLoad } from "@dcloudio/uni-app";
  64. import footerBar from "@/components/footerBar.vue";
  65. import { getRegionalGroupListAPI } from "../../api/marketSituation/marketSituation.js";
  66. import { useMarketSituationStore } from "../../stores/modules/marketSituation.js";
  67. const marketSituationStore = useMarketSituationStore();
  68. // 响应式数据
  69. const iSMT = ref(0);
  70. const contentHeight = ref(0);
  71. const headerHeight = ref(80);
  72. const marketTitle = ref();
  73. const sortType = ref(""); // 排序类型:'price' 或 'change'
  74. const sortOrder = ref("desc"); // 排序顺序:'asc' 或 'desc'
  75. const regionalGroupArray = ref([]);
  76. // 计算属性
  77. const contentTopPosition = computed(() => {
  78. return iSMT.value + headerHeight.value;
  79. });
  80. const sortedStockList = computed(() => {
  81. console.log("计算sortedStockList,原始数据长度:", marketSituationStore.marketDetailCardData.length);
  82. let list = [...marketSituationStore.marketDetailCardData];
  83. if (sortType.value === "price") {
  84. list.sort((a, b) => {
  85. return sortOrder.value === "asc" ? a.price - b.price : b.price - a.price;
  86. });
  87. } else if (sortType.value === "change") {
  88. list.sort((a, b) => {
  89. const aChange = parseFloat(a.changePercent.replace(/[+%-]/g, ""));
  90. const bChange = parseFloat(b.changePercent.replace(/[+%-]/g, ""));
  91. return sortOrder.value === "asc" ? aChange - bChange : bChange - aChange;
  92. });
  93. }
  94. console.log("排序后数据长度:", list.length);
  95. return list;
  96. });
  97. const judgeSymbol = (num) => {
  98. return num[0] === "-" ? num : "+" + num;
  99. };
  100. const getRegionalGroupList = async () => {
  101. try {
  102. const result = await getRegionalGroupListAPI({
  103. name: marketTitle.value,
  104. });
  105. regionalGroupArray.value = result.data;
  106. marketSituationStore.marketDetailCardData = result.data;
  107. } catch (e) {
  108. console.error("获取区域分组列表失败:", e);
  109. }
  110. };
  111. // 方法
  112. const goBack = () => {
  113. uni.navigateBack();
  114. };
  115. // 方法:查看指数详情
  116. const viewIndexDetail = (item, index) => {
  117. console.log("查看指数详情:", item.stockName);
  118. // 这里可以跳转到具体的指数详情页面
  119. uni.navigateTo({
  120. url: `/pages/marketSituation/marketCondition?stockInformation=${encodeURIComponent(JSON.stringify(item))}&index=${index}&from=marketDetail`,
  121. });
  122. };
  123. const sortByPrice = () => {
  124. if (sortType.value === "price") {
  125. sortOrder.value = sortOrder.value === "asc" ? "desc" : "asc";
  126. } else {
  127. sortType.value = "price";
  128. sortOrder.value = "desc";
  129. }
  130. };
  131. const sortByChange = () => {
  132. if (sortType.value === "change") {
  133. sortOrder.value = sortOrder.value === "asc" ? "desc" : "asc";
  134. } else {
  135. sortType.value = "change";
  136. sortOrder.value = "desc";
  137. }
  138. };
  139. // TCP相关响应式变量
  140. import tcpConnection, { TCPConnection, TCP_CONFIG } from "@/api/tcpConnection.js";
  141. const tcpConnected = ref(false);
  142. const connectionListener = ref(null);
  143. const messageListener = ref(null);
  144. // 初始化TCP监听器
  145. const initTcpListeners = () => {
  146. // 创建连接状态监听器并保存引用
  147. connectionListener.value = (status, result) => {
  148. tcpConnected.value = status === "connected";
  149. console.log("TCP连接状态变化:", status, tcpConnected.value);
  150. // 如果连接,发送获取批量数据
  151. if (status === "connected") {
  152. sendTcpMessage("batch_real_time");
  153. }
  154. };
  155. // 创建消息监听器并保存引用
  156. messageListener.value = (type, message, parsedArray) => {
  157. const messageObj = {
  158. type: type,
  159. content: message,
  160. parsedArray: parsedArray,
  161. timestamp: new Date().toLocaleTimeString(),
  162. direction: "received",
  163. };
  164. // 解析股票数据
  165. parseStockData(message);
  166. };
  167. // 注册监听器
  168. tcpConnection.onConnectionChange(connectionListener.value);
  169. tcpConnection.onMessage(messageListener.value);
  170. };
  171. // 连接TCP服务器
  172. const connectTcp = () => {
  173. console.log("开始连接TCP服务器...");
  174. tcpConnection.connect();
  175. };
  176. // 断开TCP连接
  177. const disconnectTcp = () => {
  178. console.log("断开TCP连接...");
  179. tcpConnection.disconnect();
  180. tcpConnected.value = false;
  181. };
  182. // 发送TCP消息
  183. const sendTcpMessage = (command) => {
  184. let messageData;
  185. let messageDataArray = [];
  186. if (command == "batch_real_time") {
  187. messageDataArray = regionalGroupArray.value.map((item) => item.code);
  188. }
  189. console.log(messageDataArray);
  190. switch (command) {
  191. // 实时行情推送
  192. case "real_time":
  193. messageData = {
  194. command: "real_time",
  195. stock_code: "SH.000001",
  196. };
  197. break;
  198. // 初始化获取行情历史数据
  199. case "init_real_time":
  200. messageData = {
  201. command: "init_real_time",
  202. stock_code: "SH.000001",
  203. };
  204. break;
  205. case "stop_real_time":
  206. messageData = {
  207. command: "stop_real_time",
  208. };
  209. break;
  210. // 股票列表
  211. case "stock_list":
  212. messageData = {
  213. command: "stock_list",
  214. };
  215. break;
  216. case "batch_real_time":
  217. messageData = {
  218. command: "batch_real_time",
  219. stock_codes: messageDataArray,
  220. };
  221. break;
  222. case "help":
  223. messageData = {
  224. command: "help",
  225. };
  226. break;
  227. }
  228. if (!messageData) {
  229. return;
  230. } else {
  231. try {
  232. // 发送消息
  233. const success = tcpConnection.send(messageData);
  234. if (success) {
  235. console.log("home发送TCP消息:", messageData);
  236. }
  237. } catch (error) {
  238. console.error("发送TCP消息时出错:", error);
  239. }
  240. }
  241. };
  242. // 获取TCP连接状态
  243. const getTcpStatus = () => {
  244. const status = tcpConnection.getConnectionStatus();
  245. uni.showModal({
  246. title: "TCP连接状态",
  247. content: `当前状态: ${status ? "已连接" : "未连接"}`,
  248. showCancel: false,
  249. });
  250. };
  251. let isMorePacket = {
  252. init_batch_real_time: false,
  253. batch_real_time: false,
  254. };
  255. let receivedMessage;
  256. // 解析TCP股票数据
  257. const parseStockData = (message) => {
  258. try {
  259. console.log("进入parseStockData, message类型:", typeof message);
  260. let parsedMessage;
  261. // 如果isMorePacket是true,说明正在接受分包数据,无条件接收
  262. // 如果message是字符串且以{开头,说明是JSON字符串,需要解析
  263. // 如果不属于以上两种情况,说明是普通字符串,不预解析
  264. if (message.includes("欢迎连接到股票数据服务器")) {
  265. console.log("服务器命令列表,不予处理");
  266. return;
  267. }
  268. if ((typeof message === "string" && message.includes("batch_data_start")) || isMorePacket.init_batch_real_time) {
  269. if (typeof message === "string" && message.includes("batch_data_start")) {
  270. console.log("开始接受分包数据");
  271. receivedMessage = "";
  272. } else {
  273. console.log("接收分包数据过程中");
  274. }
  275. isMorePacket.init_batch_real_time = true;
  276. receivedMessage += message;
  277. // 如果当前消息包含},说明收到JSON字符串结尾,结束接收,开始解析
  278. if (receivedMessage.includes("batch_data_complete")) {
  279. console.log("接受分包数据结束");
  280. isMorePacket.init_batch_real_time = false;
  281. console.log("展示数据", receivedMessage);
  282. let startIndex = 0;
  283. let startCount = 0;
  284. let endIndex = receivedMessage.indexOf("batch_data_complete");
  285. for (let i = 0; i < receivedMessage.length; ++i) {
  286. if (receivedMessage[i] == "{") {
  287. startCount++;
  288. if (startCount == 2) {
  289. startIndex = i;
  290. break;
  291. }
  292. }
  293. }
  294. for (let i = receivedMessage.indexOf("batch_data_complete"); i >= 0; --i) {
  295. if (receivedMessage[i] == "}" || startIndex == endIndex) {
  296. endIndex = i;
  297. break;
  298. }
  299. }
  300. if (startIndex >= endIndex) {
  301. throw new Error("JSON字符串格式错误");
  302. }
  303. console.log("message", startIndex, endIndex, receivedMessage[endIndex], receivedMessage[startIndex]);
  304. parsedMessage = JSON.parse(receivedMessage.substring(startIndex, endIndex + 1));
  305. console.log("JSON解析成功,解析后类型:", typeof parsedMessage, parsedMessage);
  306. const stockDataArray = parsedMessage.data;
  307. marketSituationStore.marketDetailCardData = regionalGroupArray.value.map((item) => ({
  308. market: item.market,
  309. stockCode: item.code,
  310. stockName: item.name,
  311. id: item.id,
  312. currentPrice: stockDataArray[item.code][0].current_price.toFixed(2),
  313. changeAmount: (stockDataArray[item.code][0].current_price - stockDataArray[item.code][0].pre_close).toFixed(2),
  314. changePercent: ((100 * (stockDataArray[item.code][0].current_price - stockDataArray[item.code][0].pre_close)) / stockDataArray[item.code][0].pre_close).toFixed(2) + "%",
  315. isRising: stockDataArray[item.code][0].current_price - stockDataArray[item.code][0].pre_close >= 0,
  316. }));
  317. }
  318. } else if ((typeof message === "string" && message.includes('{"count')) || isMorePacket.batch_real_time) {
  319. if (typeof message === "string" && message.includes('{"count')) {
  320. console.log("开始接受分包数据");
  321. receivedMessage = "";
  322. } else {
  323. console.log("接收分包数据过程中");
  324. }
  325. isMorePacket.batch_real_time = true;
  326. receivedMessage += message;
  327. // 如果当前消息包含},说明收到JSON字符串结尾,结束接收,开始解析
  328. if (receivedMessage.includes("batch_realtime_data")) {
  329. console.log("接受分包数据结束");
  330. isMorePacket.batch_real_time = false;
  331. console.log("展示数据", receivedMessage);
  332. let startIndex = 0;
  333. let endIndex = receivedMessage.length - 1;
  334. for (let i = 0; i < receivedMessage.length; ++i) {
  335. if (receivedMessage[i] == "{") {
  336. startIndex = i;
  337. break;
  338. }
  339. }
  340. for (let i = receivedMessage.length - 1; i >= 0; --i) {
  341. if (receivedMessage[i] == "}" || startIndex == endIndex) {
  342. endIndex = i;
  343. break;
  344. }
  345. }
  346. if (startIndex >= endIndex) {
  347. throw new Error("JSON字符串格式错误");
  348. }
  349. parsedMessage = JSON.parse(receivedMessage.substring(startIndex, endIndex + 1));
  350. console.log("JSON解析成功,解析后类型:", typeof parsedMessage, parsedMessage);
  351. const stockDataArray = parsedMessage.data;
  352. marketSituationStore.marketDetailCardData = regionalGroupArray.value.map((item) => ({
  353. market: item.market,
  354. stockCode: item.code,
  355. stockName: item.name,
  356. id: item.id,
  357. currentPrice: stockDataArray[item.code][0].current_price.toFixed(2),
  358. changeAmount: (stockDataArray[item.code][0].current_price - stockDataArray[item.code][0].pre_close).toFixed(2),
  359. changePercent: ((100 * (stockDataArray[item.code][0].current_price - stockDataArray[item.code][0].pre_close)) / stockDataArray[item.code][0].pre_close).toFixed(2) + "%",
  360. isRising: stockDataArray[item.code][0].current_price - stockDataArray[item.code][0].pre_close >= 0,
  361. }));
  362. }
  363. } else {
  364. // 没有通过JSON解析判断,说明不是需要的数据
  365. console.log("不是需要的数据,不做处理");
  366. }
  367. } catch (error) {
  368. console.error("解析TCP股票数据失败:", error.message);
  369. console.error("错误详情:", error);
  370. }
  371. };
  372. // 移除TCP监听器
  373. const removeTcpListeners = () => {
  374. if (connectionListener.value) {
  375. tcpConnection.removeConnectionListener(connectionListener.value);
  376. connectionListener.value = null;
  377. console.log("已移除TCP连接状态监听器");
  378. }
  379. if (messageListener.value) {
  380. tcpConnection.removeMessageListener(messageListener.value);
  381. messageListener.value = null;
  382. console.log("已移除TCP消息监听器");
  383. }
  384. };
  385. const startTcp = () => {
  386. try {
  387. removeTcpListeners();
  388. disconnectTcp();
  389. initTcpListeners();
  390. connectTcp();
  391. } catch (error) {
  392. console.error("建立连接并设置监听出错:", error);
  393. }
  394. };
  395. // 页面加载时接收参数
  396. onLoad(async (options) => {
  397. if (options && options.market) {
  398. marketTitle.value = options.market;
  399. await getRegionalGroupList();
  400. initTcpListeners();
  401. await nextTick();
  402. // 开始连接
  403. startTcp();
  404. }
  405. });
  406. onUnmounted(() => {
  407. sendTcpMessage("stop_real_time");
  408. removeTcpListeners();
  409. disconnectTcp();
  410. });
  411. onMounted(() => {
  412. // 获取状态栏高度
  413. iSMT.value = uni.getSystemInfoSync().statusBarHeight;
  414. // 动态计算header实际高度
  415. uni
  416. .createSelectorQuery()
  417. .select(".header_fixed")
  418. .boundingClientRect((rect) => {
  419. if (rect) {
  420. headerHeight.value = rect.height;
  421. console.log("Header实际高度:", headerHeight.value, "px");
  422. }
  423. })
  424. .exec();
  425. });
  426. // 监听headerHeight变化,重新计算contentHeight
  427. watch(headerHeight, (newHeight) => {
  428. if (newHeight > 0) {
  429. const systemInfo = uni.getSystemInfoSync();
  430. const windowHeight = systemInfo.windowHeight;
  431. const statusBarHeight = systemInfo.statusBarHeight || 0;
  432. const footerHeight = 100;
  433. contentHeight.value = windowHeight - statusBarHeight - newHeight - footerHeight;
  434. console.log("重新计算contentHeight:", contentHeight.value);
  435. }
  436. });
  437. </script>
  438. <style scoped>
  439. .main {
  440. width: 100%;
  441. height: 100vh;
  442. background-color: #f5f5f5;
  443. position: relative;
  444. }
  445. /* 自定义导航栏 */
  446. .header_fixed {
  447. position: fixed;
  448. top: 0;
  449. left: 0;
  450. right: 0;
  451. z-index: 1000;
  452. background-color: #ffffff;
  453. border-bottom: 1px solid #f0f0f0;
  454. }
  455. .header-content {
  456. display: flex;
  457. align-items: center;
  458. justify-content: space-between;
  459. height: 44px;
  460. padding: 0 15px;
  461. }
  462. .header-left,
  463. .header-right {
  464. width: 60px;
  465. display: flex;
  466. align-items: center;
  467. }
  468. .header-left {
  469. justify-content: flex-start;
  470. }
  471. .header-right {
  472. justify-content: flex-end;
  473. gap: 10px;
  474. }
  475. .back-text {
  476. font-size: 24px;
  477. color: #333333;
  478. font-weight: 500;
  479. line-height: 1;
  480. }
  481. .header-center {
  482. flex: 1;
  483. display: flex;
  484. align-items: center;
  485. justify-content: center;
  486. }
  487. .header-title {
  488. font-size: 18px;
  489. font-weight: 600;
  490. color: #333333;
  491. }
  492. .header-icon {
  493. width: 20px;
  494. height: 20px;
  495. }
  496. .more-text {
  497. font-size: 20px;
  498. color: #666666;
  499. font-weight: bold;
  500. }
  501. /* 内容区域 */
  502. .content {
  503. position: fixed;
  504. left: 0;
  505. right: 0;
  506. bottom: 0;
  507. background-color: #ffffff;
  508. }
  509. /* 表头样式 */
  510. .table-header {
  511. display: flex;
  512. padding: 12px 15px;
  513. background-color: #f8f9fa;
  514. border-bottom: 1px solid #e9ecef;
  515. }
  516. .header-item {
  517. display: flex;
  518. align-items: center;
  519. justify-content: center;
  520. cursor: pointer;
  521. }
  522. .header-item.name-column {
  523. flex: 2;
  524. justify-content: flex-start;
  525. }
  526. .header-item.price-column,
  527. .header-item.change-column {
  528. flex: 1;
  529. justify-content: center;
  530. }
  531. .header-text {
  532. font-size: 14px;
  533. color: #666666;
  534. font-weight: 500;
  535. }
  536. .sort-icon {
  537. margin-left: 4px;
  538. font-size: 12px;
  539. color: #999999;
  540. }
  541. /* 股票列表 */
  542. .stock-list {
  543. background-color: #ffffff;
  544. }
  545. .stock-row {
  546. display: flex;
  547. align-items: center;
  548. padding: 12px 15px;
  549. border-bottom: 1px solid #f5f5f5;
  550. }
  551. .stock-row:active {
  552. background-color: #f8f8f8;
  553. }
  554. .stock-cell {
  555. display: flex;
  556. flex-direction: column;
  557. align-items: center;
  558. }
  559. .stock-cell.name-column {
  560. flex: 2;
  561. align-items: flex-start;
  562. }
  563. .stock-cell.price-column,
  564. .stock-cell.change-column {
  565. flex: 1;
  566. align-items: center;
  567. }
  568. .stock-name {
  569. font-size: 15px;
  570. color: #333333;
  571. font-weight: 500;
  572. line-height: 1.2;
  573. margin-bottom: 2px;
  574. }
  575. .stock-code {
  576. font-size: 11px;
  577. color: #999999;
  578. line-height: 1.2;
  579. }
  580. .stock-price {
  581. font-size: 15px;
  582. font-weight: 600;
  583. line-height: 1.2;
  584. }
  585. .stock-change {
  586. font-size: 13px;
  587. font-weight: 500;
  588. line-height: 1.2;
  589. }
  590. .rising {
  591. color: #00c851;
  592. }
  593. .falling {
  594. color: #ff4444;
  595. }
  596. /* 底部安全区域 */
  597. /* .bottom-safe-area {
  598. height: 20px;
  599. } */
  600. /* 底部导航栏 */
  601. /* .static-footer {
  602. position: fixed;
  603. bottom: 0;
  604. left: 0;
  605. right: 0;
  606. z-index: 1000;
  607. } */
  608. </style>