import React, { useRef } from 'react'; import { View, Text, Image, StyleSheet, Animated, PanResponder, Dimensions, Platform, } from 'react-native'; import { useRouter } from 'expo-router'; import { useSafeAreaInsets } from 'react-native-safe-area-context'; import { Ionicons } from '@expo/vector-icons'; import { useLiveStore } from '../store/liveStore'; import { useVideoStore } from '../store/videoStore'; import { proxyImageUrl } from '../utils/imageUrl'; const MINI_W = 160; const MINI_H = 90; const LIVE_HEADERS = {}; function snapRelease( pan: Animated.ValueXY, curX: number, curY: number, sw: number, sh: number, ) { const snapRight = 0; const snapLeft = -(sw - MINI_W - 24); const snapX = curX < snapLeft / 2 ? snapLeft : snapRight; const clampedY = Math.max(-sh + MINI_H + 60, Math.min(60, curY)); Animated.spring(pan, { toValue: { x: snapX, y: clampedY }, useNativeDriver: false, tension: 120, friction: 10, }).start(); } export function LiveMiniPlayer() { const { isActive, roomId, title, cover, hlsUrl, clearLive } = useLiveStore(); const videoMiniActive = useVideoStore(s => s.isActive); const router = useRouter(); const insets = useSafeAreaInsets(); const pan = useRef(new Animated.ValueXY()).current; const isDragging = useRef(false); // 用 ref 保持最新值,避免 PanResponder 闭包捕获过期的初始值 const storeRef = useRef({ roomId, clearLive, router }); storeRef.current = { roomId, clearLive, router }; const panResponder = useRef( PanResponder.create({ onStartShouldSetPanResponder: () => true, onPanResponderGrant: () => { isDragging.current = false; pan.setOffset({ x: (pan.x as any)._value, y: (pan.y as any)._value }); pan.setValue({ x: 0, y: 0 }); }, onPanResponderMove: (_, gs) => { if (Math.abs(gs.dx) > 5 || Math.abs(gs.dy) > 5) { isDragging.current = true; } pan.x.setValue(gs.dx); pan.y.setValue(gs.dy); }, onPanResponderRelease: (evt) => { pan.flattenOffset(); if (!isDragging.current) { const { locationX, locationY } = evt.nativeEvent; const { roomId: rid, clearLive: clear, router: r } = storeRef.current; if (locationX > MINI_W - 28 && locationY < 28) { clear(); } else { r.push(`/live/${rid}` as any); } return; } const { width: sw, height: sh } = Dimensions.get('window'); snapRelease(pan, (pan.x as any)._value, (pan.y as any)._value, sw, sh); }, onPanResponderTerminate: () => { pan.flattenOffset(); }, }), ).current; if (!isActive) return null; const bottomOffset = insets.bottom + 16 + (videoMiniActive ? 106 : 0); // Web 端降级:封面图 + LIVE 徽标 if (Platform.OS === 'web') { return ( LIVE {title} ); } // Native:实际 HLS 流播放 const Video = require('react-native-video').default; return ( {/* pointerEvents="none" 防止 Video 原生层吞噬触摸事件 */} LIVE {title} {/* 关闭按钮视觉层,点击逻辑由 onPanResponderRelease 坐标判断 */} ); } const styles = StyleSheet.create({ container: { position: 'absolute', right: 12, width: MINI_W, height: MINI_H, borderRadius: 8, backgroundColor: '#1a1a1a', overflow: 'hidden', elevation: 8, shadowColor: '#000', shadowOffset: { width: 0, height: 2 }, shadowOpacity: 0.3, shadowRadius: 4, }, videoArea: { width: '100%', height: 66, backgroundColor: '#111', }, liveBadge: { position: 'absolute', top: 4, left: 6, flexDirection: 'row', alignItems: 'center', backgroundColor: 'rgba(0,0,0,0.55)', paddingHorizontal: 5, paddingVertical: 2, borderRadius: 3, gap: 3, }, liveDot: { width: 5, height: 5, borderRadius: 2.5, backgroundColor: '#f00' }, liveText: { color: '#fff', fontSize: 9, fontWeight: '700', letterSpacing: 0.5 }, titleText: { color: '#fff', fontSize: 11, paddingHorizontal: 6, paddingVertical: 3, lineHeight: 14, height: 24, backgroundColor: '#1a1a1a', }, closeBtn: { position: 'absolute', top: 4, right: 4, width: 18, height: 18, borderRadius: 9, backgroundColor: 'rgba(0,0,0,0.6)', alignItems: 'center', justifyContent: 'center', }, });