zdl 95134d526d feat(HistoricalEvents): 优化历史事件列表 UI 和相关股票弹窗
主要改进:
1. 历史事件列表改为卡片式网格布局
   - 移除时间轴样式(垂直线 + 节点图标)
   - 使用 SimpleGrid 响应式布局(1列/2列/3列)
   - 卡片显示:事件名称、日期、相关度、重要性、描述
   - 点击"相关股票"按钮打开 Modal 弹窗

2. 历史事件对比默认展开
   - DynamicNewsDetailPanel: isHistoricalOpen 初始值改为 true
   - 用户打开事件详情面板时,历史事件对比区域默认展开

3. 相关股票弹窗改为卡片式布局
   - StocksList 组件从 Table 表格改为 SimpleGrid 卡片
   - 显示 6 个字段:代码、名称、板块、相关度、涨幅、关联原因
   - 关联原因支持展开/收起(startingHeight: 40px)
   - 响应式网格布局(base: 1列, md: 2列, lg: 3列)

4. 修复字段映射兼容性
   - 添加 getEventDate() 兼容多种日期字段
   - 添加 getEventContent() 兼容多种内容字段
   - 支持字段:event_date/created_at/date、content/description/summary
   - 添加 Debug 日志输出实际数据结构

5. 修复弹窗关闭问题
   - 添加 handleCloseModal() 同时清空两个状态
   - 使用条件渲染 {stocksModalOpen && <Modal>}
   - 关闭时完全卸载 Modal 组件,避免状态残留

技术细节:
- 移除未使用的导入(Table, Thead, Tbody, Tr, Th, Td 等)
- 新增工具函数:formatChange, getChangeColor, getCorrelationColor
- 卡片 hover 效果:boxShadow + borderColor 变化
- 涨跌幅颜色:红色(上涨)/ 绿色(下跌)
- 相关度颜色梯度:>=80% 红色, >=60% 橙色, <60% 绿色

代码统计:
- HistoricalEvents.js: -402 行, +344 行(净减少 58 行)
- 移除时间轴复杂逻辑,简化组件结构
- 提升代码可维护性和可读性

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-03 12:41:02 +08:00
2025-10-11 12:02:01 +08:00
2025-10-29 16:19:01 +08:00
2025-10-29 16:00:21 +08:00
2025-10-11 12:02:01 +08:00
2025-10-11 12:02:01 +08:00
2025-10-29 16:27:31 +08:00
2025-10-11 12:02:01 +08:00
2025-10-11 12:02:01 +08:00
2025-10-13 19:53:13 +08:00
2025-10-11 12:02:01 +08:00
2025-10-11 12:02:01 +08:00
2025-10-30 15:42:54 +08:00
2025-10-15 11:49:55 +08:00
2025-10-13 19:53:13 +08:00
2025-10-31 10:33:53 +08:00
2025-10-11 12:02:01 +08:00
2025-10-13 19:53:13 +08:00
2025-10-11 12:02:01 +08:00
2025-10-11 16:16:02 +08:00
2025-10-11 12:02:01 +08:00
2025-10-11 12:02:01 +08:00
2025-10-11 12:02:01 +08:00

vf_react

前端


📚 重构记录

2025-10-30: EventList.js 组件化重构

🎯 重构目标

将 Community 社区页面的 EventList.js 组件1095行拆分为多个可复用的子组件提高代码可维护性和复用性。

📊 重构成果

  • 重构前: 1095 行
  • 重构后: 497 行
  • 减少: 598 行 (-54.6%)

📁 新增目录结构

src/views/Community/components/EventCard/
├── index.js                      (60行)  - EventCard 统一入口,智能路由紧凑/详细模式
│
├── ──────────────────────────────────────────────────────────
│   原子组件 (Atoms) - 7个基础UI组件
├── ──────────────────────────────────────────────────────────
│
├── EventTimeline.js              (60行)  - 时间轴显示组件
│   └── Props: createdAt, timelineStyle, borderColor, minHeight
│
├── EventImportanceBadge.js       (100行) - 重要性等级标签 (S/A/B/C/D)
│   └── Props: importance, showTooltip, showIcon, size
│
├── EventStats.js                 (60行)  - 统计信息 (浏览/帖子/关注)
│   └── Props: viewCount, postCount, followerCount, size, spacing
│
├── EventFollowButton.js          (40行)  - 关注按钮
│   └── Props: isFollowing, followerCount, onToggle, size, showCount
│
├── EventPriceDisplay.js          (130行) - 价格变动显示 (平均/最大/周)
│   └── Props: avgChange, maxChange, weekChange, compact, inline
│
├── EventDescription.js           (60行)  - 描述文本 (支持展开/收起)
│   └── Props: description, textColor, minLength, noOfLines
│
├── EventHeader.js                (100行) - 事件标题头部
│   └── Props: title, importance, onTitleClick, linkColor, compact
│
├── ──────────────────────────────────────────────────────────
│   组合组件 (Molecules) - 2个卡片组件
├── ──────────────────────────────────────────────────────────
│
├── CompactEventCard.js           (160行) - 紧凑模式事件卡片
│   ├── 使用: EventTimeline, EventHeader, EventStats, EventFollowButton
│   └── Props: event, index, isFollowing, followerCount, callbacks...
│
└── DetailedEventCard.js          (170行) - 详细模式事件卡片
    ├── 使用: EventTimeline, EventHeader, EventStats, EventFollowButton,
    │        EventPriceDisplay, EventDescription
    └── Props: event, isFollowing, followerCount, callbacks...

总计: 10个文件940行代码


🔧 重构的文件

src/views/Community/components/EventList.js

移除的内容:

  • renderPriceChange 函数 (~60行)
  • renderCompactEvent 函数 (~200行)
  • renderDetailedEvent 函数 (~300行)
  • expandedDescriptions state展开状态管理移至子组件
  • 冗余的 Chakra UI 导入

保留的功能:

  • WebSocket 实时推送
  • 浏览器原生通知
  • 关注状态管理 (followingMap, followCountMap)
  • 分页控制
  • 视图模式切换(紧凑/详细)
  • 推送权限管理

新增引入:

import EventCard from './EventCard';

🏗️ 架构改进

重构前(单体架构)

EventList.js (1095行)
├── 业务逻辑 (WebSocket, 关注, 通知)
├── renderCompactEvent (200行)
│   └── 所有UI代码内联
├── renderDetailedEvent (300行)
│   └── 所有UI代码内联
└── renderPriceChange (60行)

重构后(组件化架构)

EventList.js (497行) - 容器组件
├── 业务逻辑 (WebSocket, 关注, 通知)
└── 渲染逻辑
    └── EventCard (智能路由)
        ├── CompactEventCard (紧凑模式)
        │   ├── EventTimeline
        │   ├── EventHeader (compact)
        │   ├── EventStats
        │   └── EventFollowButton
        └── DetailedEventCard (详细模式)
            ├── EventTimeline
            ├── EventHeader (detailed)
            ├── EventStats
            ├── EventFollowButton
            ├── EventPriceDisplay
            └── EventDescription

优势

  1. 可维护性 ⬆️

    • 每个组件职责单一(单一职责原则)
    • 代码行数减少 54.6%
    • 组件边界清晰,易于理解
  2. 可复用性 ⬆️

    • 原子组件可在其他页面复用
    • 例如EventImportanceBadge 可用于任何需要显示事件等级的地方
  3. 可测试性 ⬆️

    • 小组件更容易编写单元测试
    • 可独立测试每个组件的渲染和交互
  4. 性能优化 ⬆️

    • React 可以更精确地追踪变化
    • 减少不必要的重渲染
    • 每个子组件可独立优化useMemo, React.memo
  5. 开发效率 ⬆️

    • 新增功能时只需修改对应的子组件
    • 代码审查更高效
    • 降低了代码冲突的概率

📦 依赖工具函数

本次重构使用了之前提取的工具函数:

src/utils/priceFormatters.js          (105行)
├── getPriceChangeColor(value)        - 获取价格变化文字颜色
├── getPriceChangeBg(value)           - 获取价格变化背景颜色
├── getPriceChangeBorderColor(value)  - 获取价格变化边框颜色
├── formatPriceChange(value)          - 格式化价格为字符串
└── PriceArrow({ value })             - 价格涨跌箭头组件

src/constants/animations.js           (72行)
├── pulseAnimation                    - 脉冲动画S/A级标签
├── fadeIn                            - 渐入动画
├── slideInUp                         - 从下往上滑入
├── scaleIn                           - 缩放进入
└── spin                              - 旋转动画Loading

🚀 下一步优化计划

Phase 1 已完成,后续可继续优化:

  • Phase 2: 拆分 StockDetailPanel.js (1067行 → ~250行)
  • Phase 3: 拆分 InvestmentCalendar.js (827行 → ~200行)
  • Phase 4: 拆分 MidjourneyHeroSection.js (813行 → ~200行)
  • Phase 5: 拆分 UnifiedSearchBox.js (679行 → ~180行)

🔗 相关提交

  • feat: 拆分 EventList.js/提取价格相关工具函数到 utils/priceFormatters.js
  • feat(EventList): 创建事件卡片原子组件
  • feat(EventList): 创建事件卡片组合组件
  • refactor(EventList): 使用组件化架构替换内联渲染函数

Description
前端
Readme 195 MiB
Languages
HTML 75.9%
JavaScript 9.6%
CSS 8%
SCSS 3.2%
Python 2.1%
Other 1.2%