eebd20727637dec811079292ac409cb2c406d3da
## 问题 生产环境收到 WebSocket 消息但不触发通知: - Network 面板显示消息已接收 - 但事件监听器未触发(事件处理函数不执行) - 手动测试 `window.__TEST_NOTIFICATION__.testAllTypes()` 正常工作 - 诊断脚本显示 `window.socket: undefined` ## 根本原因 Socket 实例未暴露到全局作用域,导致: 1. 无法验证 NotificationContext 中的监听器是否注册在正确的 Socket 实例上 2. 可能存在多个 Socket 实例(导入的实例 vs 实际连接的实例) 3. 事件监听器注册在错误的实例上 ## 解决方案 在 `src/services/socket/index.js` 中暴露 Socket 实例到 window 对象: ### 代码变更 ```javascript // ⚡ 新增:暴露 Socket 实例到 window(用于调试和验证) if (typeof window !== 'undefined') { window.socket = socketService; window.socketService = socketService; console.log('✅ Socket instance exposed to window'); console.log(' 📍 window.socket:', window.socket); console.log(' 📍 Socket.IO instance:', window.socket?.socket); console.log(' 📍 Connection status:', window.socket?.connected); } ``` ## 好处 1. **可调试性**: 可在浏览器 Console 直接访问 Socket 实例 2. **验证监听器**: 可检查 `window.socket.socket._callbacks` 确认监听器已注册 3. **诊断连接**: 可实时查看 `window.socket.connected` 状态 4. **手动测试**: 可通过 `window.socket.emit()` 手动触发事件 ## 验证步骤 部署后在浏览器 Console 执行: ```javascript // 1. 验证 Socket 实例已暴露 console.log(window.socket); // 2. 检查连接状态 console.log('Connected:', window.socket.connected); // 3. 检查监听器 console.log('Listeners:', window.socket.socket._callbacks); // 4. 测试手动触发事件 window.socket.socket.emit('new_event', { id: 999, title: 'Test' }); ``` ## 影响范围 - 修改文件: `src/services/socket/index.js`(1 个文件) - 影响范围: 仅新增调试功能,不改变业务逻辑 - 风险等级: 低(只读暴露,不修改 Socket 行为) 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
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行) - ❌
expandedDescriptionsstate(展开状态管理移至子组件) - ❌ 冗余的 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
✨ 优势
-
可维护性 ⬆️
- 每个组件职责单一(单一职责原则)
- 代码行数减少 54.6%
- 组件边界清晰,易于理解
-
可复用性 ⬆️
- 原子组件可在其他页面复用
- 例如:EventImportanceBadge 可用于任何需要显示事件等级的地方
-
可测试性 ⬆️
- 小组件更容易编写单元测试
- 可独立测试每个组件的渲染和交互
-
性能优化 ⬆️
- React 可以更精确地追踪变化
- 减少不必要的重渲染
- 每个子组件可独立优化(useMemo, React.memo)
-
开发效率 ⬆️
- 新增功能时只需修改对应的子组件
- 代码审查更高效
- 降低了代码冲突的概率
📦 依赖工具函数
本次重构使用了之前提取的工具函数:
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.jsfeat(EventList): 创建事件卡片原子组件feat(EventList): 创建事件卡片组合组件refactor(EventList): 使用组件化架构替换内联渲染函数
Description
Languages
HTML
75.9%
JavaScript
9.6%
CSS
8%
SCSS
3.2%
Python
2.1%
Other
1.2%