mirror of
https://github.com/tiajinsha/JKVideo.git
synced 2026-04-04 22:49:02 +08:00
203 lines
5.8 KiB
TypeScript
203 lines
5.8 KiB
TypeScript
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 (
|
||
<Animated.View
|
||
style={[styles.container, { bottom: bottomOffset, transform: pan.getTranslateTransform() }]}
|
||
{...panResponder.panHandlers}
|
||
>
|
||
<Image source={{ uri: proxyImageUrl(cover) }} style={styles.videoArea} />
|
||
<View style={styles.liveBadge} pointerEvents="none">
|
||
<View style={styles.liveDot} />
|
||
<Text style={styles.liveText}>LIVE</Text>
|
||
</View>
|
||
<Text style={styles.titleText} numberOfLines={1}>{title}</Text>
|
||
<View style={styles.closeBtn}>
|
||
<Ionicons name="close" size={14} color="#fff" />
|
||
</View>
|
||
</Animated.View>
|
||
);
|
||
}
|
||
|
||
// Native:实际 HLS 流播放
|
||
const Video = require('react-native-video').default;
|
||
|
||
return (
|
||
<Animated.View
|
||
style={[styles.container, { bottom: bottomOffset, transform: pan.getTranslateTransform() }]}
|
||
{...panResponder.panHandlers}
|
||
>
|
||
{/* pointerEvents="none" 防止 Video 原生层吞噬触摸事件 */}
|
||
<View style={styles.videoArea} pointerEvents="none">
|
||
<Video
|
||
key={hlsUrl}
|
||
source={{ uri: hlsUrl, headers: LIVE_HEADERS }}
|
||
style={StyleSheet.absoluteFill}
|
||
resizeMode="cover"
|
||
controls={false}
|
||
muted={false}
|
||
paused={false}
|
||
repeat={false}
|
||
onError={clearLive}
|
||
/>
|
||
</View>
|
||
<View style={styles.liveBadge} pointerEvents="none">
|
||
<View style={styles.liveDot} />
|
||
<Text style={styles.liveText}>LIVE</Text>
|
||
</View>
|
||
<Text style={styles.titleText} numberOfLines={1}>{title}</Text>
|
||
{/* 关闭按钮视觉层,点击逻辑由 onPanResponderRelease 坐标判断 */}
|
||
<View style={styles.closeBtn}>
|
||
<Ionicons name="close" size={14} color="#fff" />
|
||
</View>
|
||
</Animated.View>
|
||
);
|
||
}
|
||
|
||
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',
|
||
},
|
||
});
|