zdl 6b96744b2c fix(notification): 修复 Socket 重连后通知功能失效问题(方案2)
采用完全重构的方式解决 Socket 重连后事件监听器丢失和闭包陷阱问题。

## 核心问题
1. Socket 重连后,事件监听器被重复注册,导致监听器累积或丢失
2. 闭包陷阱:监听器捕获了过期的 addNotification 函数引用
3. 依赖循环:registerSocketEvents 依赖 addNotification,导致频繁重新创建

## 解决方案(方案2:完全重构)

### 1. 使用 Ref 存储最新函数引用
```javascript
const addNotificationRef = useRef(null);
const adaptEventToNotificationRef = useRef(null);
const isFirstConnect = useRef(true);
```

### 2. 同步最新函数到 Ref
通过 useEffect 确保 ref.current 始终指向最新的函数:
```javascript
useEffect(() => {
    addNotificationRef.current = addNotification;
}, [addNotification]);
```

### 3. 监听器只注册一次
- useEffect 依赖数组改为 `[]`(空数组)
- socket.on('new_event') 只在组件挂载时注册一次
- 监听器内部使用 `ref.current` 访问最新函数

### 4. 重连时只重新订阅
- Socket 重连后只调用 `subscribeToEvents()`
- 不再重新注册监听器(避免累积)

## 关键代码变更

### NotificationContext.js
- **新增 Ref 定义**(第 62-65 行):存储最新的回调函数引用
- **新增同步 useEffect**(第 607-615 行):保持 ref 与函数同步
- **删除 registerSocketEvents 函数**:不再需要提取事件注册逻辑
- **重构 Socket useEffect**(第 618-824 行):
  - 依赖数组: `[registerSocketEvents, toast]` → `[]`
  - 监听器注册: 只在初始化时执行一次
  - 重连处理: 只调用 `subscribeToEvents()`,不重新注册监听器
  - 防御性检查: 确保 ref 已初始化再使用

## 技术优势

### 彻底解决重复注册
-  监听器生命周期与组件绑定,只注册一次
-  Socket 重连不会触发监听器重新注册

### 避免闭包陷阱
-  `ref.current` 始终指向最新的函数
-  监听器不受 useEffect 依赖变化影响

### 简化依赖管理
-  useEffect 无依赖,不会因状态变化而重新运行
-  性能优化:减少不必要的函数创建和监听器操作

### 提升代码质量
-  逻辑更清晰:所有监听器集中在一个 useEffect
-  易于维护:依赖关系简单明了
-  详细日志:便于调试和追踪问题

## 验证测试

### 测试场景
1.  首次连接 + 接收事件 → 正常显示通知
2.  断开重连 + 接收事件 → 重连后正常接收通知
3.  多次重连 → 每次重连后通知功能正常
4.  控制台无重复注册警告

### 预期效果
- 首次连接: 显示 " 首次连接成功"
- 重连成功: 显示 "🔄 重连成功!" (不显示 "registerSocketEvents() 被调用")
- 收到事件: 根据页面可见性显示网页通知或浏览器通知

## 影响范围
- 修改文件: `src/contexts/NotificationContext.js`
- 影响功能: Socket 连接管理、事件监听、通知分发
- 兼容性: 完全向后兼容,无破坏性变更

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

Co-Authored-By: Claude <noreply@anthropic.com>
2025-11-11 13:35:08 +08:00
2025-10-11 12:02:01 +08:00
2025-11-10 20:05:53 +08:00
2025-10-29 16:19:01 +08:00
2025-11-11 11:45:19 +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-11-10 12:22:21 +08:00
2025-11-07 08:13:12 +08:00
2025-11-07 21:46:50 +08:00
2025-10-11 12:02:01 +08:00
2025-10-13 19:53:13 +08:00
2025-11-10 20:05:53 +08:00
2025-10-11 12:02:01 +08:00
2025-10-30 15:42:54 +08:00
2025-11-07 17:42:06 +08:00
2025-11-10 07:56:52 +08:00
2025-10-15 11:49:55 +08:00
2025-10-13 19:53:13 +08:00
2025-11-07 21:03:24 +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-11-03 16:10:35 +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%