import { View, Text, Image as NativeImage, TouchableOpacity, useWindowDimensions, Animated, Easing, Modal, } from "react-native"; import React, { useState, useEffect, useCallback, useMemo } from "react"; import { useTailwind } from "tailwind-rn"; import { useSafeAreaInsets } from "react-native-safe-area-context"; import Toast from "react-native-toast-message"; import { Image } from "expo-image"; import { useHeaderHeight } from "@react-navigation/elements"; import { TabView, SceneMap, TabBar } from "react-native-tab-view"; import BottomSheet, { BottomSheetModalProvider } from "@gorhom/bottom-sheet"; import { GestureHandlerRootView } from "react-native-gesture-handler"; import AllSpacePosts from "./AllSpacePosts"; import IronFanSpacePosts from "./IronFanSpacePosts"; import SuperFanSpacePosts from "./SuperFanSpacePosts"; import GetWechatModal from "../../components/GetWechatModal"; import SubmitWechatModal from "../../components/SubmitWechatModal"; import baseRequest from "../../utils/baseRequest"; import { generateSignature } from "../../utils/crypto"; import CreatePostModal from "../../components/CreatePostModal"; import { LinearProgress, Button } from "@rneui/themed"; import { LinearGradient } from "expo-linear-gradient"; import { usePreventScreenCapture } from "expo-screen-capture"; import { Svg, Path } from "react-native-svg"; import MyModal from "../../components/MyModal"; export default function StreamerSpace({ navigation, route }) { usePreventScreenCapture(); const blurhash = "LcKUTa%gOYWBYRt6xuoJo~s8V@fk"; const tailwind = useTailwind(); const insets = useSafeAreaInsets(); const headerHeight = useHeaderHeight(); //创建动态Modal是否展示 const [isCreatePostModalVisible, setIsCreatePostModalVisible] = useState(false); //退款中Modal是否展示 const [isRefundingModalVisible, setIsRefundingModalVisible] = useState(false); //获取空间数据并将该空间标为已读 const [data, setData] = useState({}); const getData = async () => { const apiUrl = process.env.EXPO_PUBLIC_API_URL; try { const base = await baseRequest(); const signature = await generateSignature({ mid: route.params.mid, ...base, }); const _response = await fetch( `${apiUrl}/api/zone/list_by_mid?signature=${signature}`, { method: "POST", headers: { "Content-Type": "application/json", }, body: JSON.stringify({ mid: route.params.mid, ...base, }), } ); const _data = await _response.json(); if (_data.ret === -1) { Toast.show({ type: "error", text1: _data.msg, topOffset: 60, }); return; } if (_data.data.list[0].visitor_role === 4) { navigation.replace("SpaceIntroduce", { mid: route.params.mid }); return; } setData({ ..._data.data.list[0], refund_enable: _data.data.refund_enable, refund_status: _data.data.refund_status, }); if (_data.data.refund_status === 1) { setIsRefundingModalVisible(true); } const signature2 = await generateSignature({ zid: _data.data.list[0].id, ...base, }); const _response2 = await fetch( `${apiUrl}/api/zone_session/upsert?signature=${signature2}`, { method: "POST", headers: { "Content-Type": "application/json", }, body: JSON.stringify({ zid: _data.data.list[0].id, ...base, }), } ); const _data2 = await _response2.json(); if (_data2.ret === -1) { Toast.show({ type: "error", text1: _data2.msg, topOffset: 60, }); return; } } catch (error) { console.error(error); } }; useEffect(() => { getData(); }, []); //设置header右侧按钮功能 useEffect(() => { navigation.setOptions({ headerRight: () => ( <TouchableOpacity onPress={() => navigation.navigate("SpaceSetting", { data: data })} > <NativeImage source={require("../../assets/icon/others/setting.png")} /> </TouchableOpacity> ), }); }, [data]); //点击查看微信按钮 const [isAddWechatModalVisible, setIsAddWechatModalVisible] = useState(false); //tab组件相关 const layout = useWindowDimensions(); const [index, setIndex] = useState(0); const [routes, setRoutes] = useState([ { key: "all", title: "全部" }, { key: "ironFan", title: "铁粉专享" }, { key: "superFan", title: "超粉专享" }, ]); //当主播没开通超粉功能时不展示超粉专享tab useEffect(() => { if (data?.is_superfanship_enabled === 0) setRoutes([ { key: "all", title: "全部" }, { key: "ironFan", title: "铁粉专享" }, ]); }, [data]); const renderScene = useCallback( SceneMap({ all: () => <AllSpacePosts zid={data?.id} />, ironFan: () => <IronFanSpacePosts zid={data?.id} />, superFan: () => <SuperFanSpacePosts zid={data?.id} />, }), [data] ); const renderIndicator = useCallback((props) => { const { position, navigationState, getTabWidth } = props; const inputRange = [0, 1]; const translateX = position.interpolate({ inputRange: inputRange, outputRange: inputRange.map((x) => { return x * getTabWidth(navigationState.index); }), }); return ( <Animated.View style={{ width: `${100 / navigationState.routes.length}%`, transform: [ { translateX, }, ], paddingBottom: 12, ...tailwind("flex flex-1 items-center justify-end"), }} > <NativeImage source={require("../../assets/icon/others/pinkline.png")} /> </Animated.View> ); }, []); const renderTabBar = useCallback( (props) => ( <TabBar {...props} activeColor="#FFFFFF" inactiveColor="#FFFFFF80" style={tailwind("bg-transparent")} labelStyle={tailwind("text-lg font-medium")} renderIndicator={renderIndicator} /> ), [] ); //bottom sheet组件相关 const snapPoints = useMemo(() => ["73%", "100%"], []); //刷新铁粉身份(解决主播降价导致的进度满100但未成为铁粉) const handleFansIdentityRefresh = async () => { const apiUrl = process.env.EXPO_PUBLIC_API_URL; try { const base = await baseRequest(); const body = { zid: data?.id, ...base, }; const signature = await generateSignature(body); const _response = await fetch( `${apiUrl}/api/zone_moment/fans_identity_refresh?signature=${signature}`, { method: "POST", headers: { "Content-Type": "application/json", }, body: JSON.stringify(body), } ); const _data = await _response.json(); if (_data.ret === -1) { Toast.show({ type: "error", text1: _data.msg, topOffset: 60, }); return; } await getData(); setIsIronFanModalVisible(false); } catch (error) { console.error(error); } }; //成为铁粉Modal组件 const [isIronFanModalVisible, setIsIronFanModalVisible] = useState(false); const ironFanProgress = useMemo( () => Math.floor((data?.expenditure / data?.ironfanship_price) * 100), [data] ); const BecomeIronFanModal = useCallback( () => ( <Modal visible={isIronFanModalVisible} transparent={true} statusBarTranslucent animationType="fade" > <TouchableOpacity activeOpacity={1} style={{ backgroundColor: "#00000080", ...tailwind("flex-1 justify-center items-center"), }} onPress={() => setIsIronFanModalVisible(false)} > <TouchableOpacity activeOpacity={1} style={tailwind( "flex flex-col p-4 rounded-2xl bg-[#17161A] items-center w-3/4" )} > <View style={tailwind("flex-row items-end w-full")}> <Text style={tailwind("text-base text-white font-medium")}> 当前铁粉解锁进度: </Text> <Text style={tailwind("text-2xl text-[#FF669E] font-medium")}> {`${ironFanProgress}%`} </Text> </View> <LinearProgress style={tailwind("flex-row w-full h-2 mt-2 rounded-full")} value={ironFanProgress / 100} color="#FF669E" variant="determinate" /> <Text style={tailwind("text-sm text-[#FF669E] font-medium")}>{`${ data?.expenditure / 100 } / ${data?.ironfanship_price / 100}`}</Text> <Text style={tailwind("text-sm font-medium text-[#FFFFFF80] mt-2")}> 空间内累计消费达到¥{data?.ironfanship_price / 100}即可成为 <Text style={tailwind("text-[#FF669E]")}>铁粉</Text> ,可查看所有铁粉专享内容哦,快来成为我的铁粉吧~ </Text> <Button onPress={() => { setIndex(1); setIsIronFanModalVisible(false); }} ViewComponent={LinearGradient} radius="999" size="md" linearGradientProps={{ colors: ["#FF668B", "#FF66F0"], start: { x: 0, y: 0.5 }, end: { x: 1, y: 0.5 }, }} titleStyle={tailwind("text-base font-medium")} containerStyle={tailwind("w-full px-4 mt-4")} > 查看铁粉专享内容 </Button> <Text onPress={handleFansIdentityRefresh} style={{ textDecorationLine: "underline", ...tailwind("text-sm font-medium text-[#FF669E] mt-2"), }} > 进度已满但还未成为铁粉? </Text> </TouchableOpacity> </TouchableOpacity> </Modal> ), [isIronFanModalVisible] ); //刷新动画 const [rotateValue] = useState(new Animated.Value(0)); const startRotation = () => { getData(); Animated.timing(rotateValue, { toValue: 1, duration: 500, easing: Easing.linear, useNativeDriver: true, }).start(() => { rotateValue.setValue(0); }); }; const spin = rotateValue.interpolate({ inputRange: [0, 1], outputRange: ["0deg", "360deg"], }); return ( <GestureHandlerRootView style={{ flex: 1 }}> <BottomSheetModalProvider> <View style={{ paddingLeft: insets.left, paddingRight: insets.right, ...tailwind("flex-1"), }} > <View style={tailwind("absolute top-0 left-0")}> <Image source={data?.streamer_ext?.cover?.images[0]?.urls[0]} contentFit="cover" transition={1000} placeholder={blurhash} cachePolicy="disk" style={{ aspectRatio: "15/13", ...tailwind("w-full"), }} /> <View style={{ backgroundColor: "#000000B2", ...tailwind("absolute w-full h-full"), }} ></View> </View> <View style={{ marginTop: headerHeight, ...tailwind("flex flex-col flex-1"), }} > <View style={tailwind("flex flex-row items-center px-4 h-24")}> <Image style={tailwind( "w-[4.6rem] h-[4.6rem] rounded-full border-2 border-white" )} source={data?.streamer_ext?.avatar?.images[0]?.urls[0]} placeholder={blurhash} contentFit="cover" transition={1000} cachePolicy="disk" /> <View style={tailwind("flex flex-col shrink mx-2 justify-between")} > <Text style={{ fontSize: 22, ...tailwind("text-white font-medium"), }} numberOfLines={1} ellipsizeMode="tail" > {data?.streamer_ext?.name} </Text> <View style={tailwind("flex-row flex-nowrap mt-1.5")}> <View style={tailwind( "flex flex-row items-center py-0.5 px-2 mr-2 bg-[#FFFFFF1A] rounded-full" )} > <NativeImage source={require("../../assets/icon/12DP/ID.png")} /> <Text style={tailwind("text-white text-xs font-medium ml-0.5")} > {data?.streamer_ext?.user_id} </Text> </View> <View style={tailwind( "flex flex-row items-center py-0.5 px-2 mr-2 bg-[#FFFFFF1A] rounded-full" )} > <NativeImage source={require("../../assets/icon/12DP/edit.png")} /> <Text style={tailwind("text-white text-xs font-medium ml-0.5")} > {data?.zone_moment_count} </Text> </View> </View> </View> <TouchableOpacity onPress={() => navigation.navigate("ShareSpace", { data: data }) } style={tailwind( "flex items-center justify-center ml-auto h-8 px-4 bg-[#FF669E] rounded-full" )} > <Text style={tailwind("text-white text-sm font-medium")}> 分享 </Text> </TouchableOpacity> </View> <View style={{ gap: 22, ...tailwind("flex flex-row px-6 mt-2") }}> <TouchableOpacity onPress={() => setIsAddWechatModalVisible(true)} style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/wechat_bg.png")} /> <Text style={tailwind("text-white text-xs font-medium mt-0.5")}> 查看微信 </Text> </TouchableOpacity> <TouchableOpacity onPress={() => setIsIronFanModalVisible(true)} style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/tiefen_bg.png")} /> <View style={tailwind("flex flex-col items-center")}> <Text style={tailwind("text-white text-xs font-medium mt-0.5")} > {data?.is_ironfanship_unlocked === 1 ? "已是铁粉" : "成为铁粉"} </Text> <Text style={{ fontSize: 10, ...tailwind("text-[#FFFFFF80] font-medium mt-0.5"), }} > {`${parseInt(data?.expenditure / 100, 10)}/${parseInt( data?.ironfanship_price / 100, 10 )}`} </Text> </View> </TouchableOpacity> {data?.is_superfanship_enabled === 1 && ( <TouchableOpacity onPress={ data?.is_superfanship_unlocked === 1 ? () => setIndex(2) : () => navigation.navigate("WebWithoutHeader", { uri: process.env.EXPO_PUBLIC_WEB_URL + "/zone/pay/" + data?.id + "/h5_zone_superfanship/0", }) } style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/chaofen_bg.png")} /> <Text style={tailwind("text-white text-xs font-medium mt-0.5")} > {data?.is_superfanship_unlocked === 1 ? "尊贵超粉" : "成为超粉"} </Text> </TouchableOpacity> )} <TouchableOpacity onPress={() => navigation.navigate("MessageDetail", { mid: 1, }) } style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/report_bg.png")} /> <Text style={tailwind("text-white text-xs font-medium mt-0.5")}> 举报 </Text> </TouchableOpacity> {data?.visitor_role === 3 && ( <TouchableOpacity onPress={() => navigation.navigate("VisibleToOneselfSpacePosts") } style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/review_fail_bg.png")} /> <Text style={tailwind("text-white text-xs font-medium mt-0.5")} > 审核未通过 </Text> </TouchableOpacity> )} </View> <BottomSheet snapPoints={snapPoints} backgroundStyle={tailwind("bg-[#07050A]")} handleIndicatorStyle={tailwind("bg-white")} handleStyle={{ paddingBottom: 0 }} > <TabView navigationState={{ index, routes }} swipeEnabled={false} renderScene={renderScene} renderTabBar={renderTabBar} onIndexChange={setIndex} initialLayout={{ width: layout.width }} /> </BottomSheet> </View> {data?.visitor_role === 3 ? ( <TouchableOpacity onPress={() => setIsCreatePostModalVisible(!isCreatePostModalVisible) } style={{ ...tailwind("absolute"), bottom: insets.bottom + 20, right: 16, }} > <NativeImage source={require("../../assets/icon/others/createpost.png")} /> </TouchableOpacity> ) : ( <View style={{ ...tailwind( "absolute bottom-0 flex flex-row w-full py-3 px-6 justify-around border-t border-[#FFFFFF26] bg-[#07050A]" ), height: 72 + insets.bottom, }} > <TouchableOpacity onPress={() => setIsAddWechatModalVisible(true)} style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/wechat.png")} /> <Text style={tailwind("text-white text-xs font-medium mt-0.5")}> 查看微信 </Text> </TouchableOpacity> {data?.is_superfanship_enabled === 1 && ( <TouchableOpacity onPress={ data?.is_superfanship_unlocked === 1 ? () => setIndex(2) : () => navigation.navigate("WebWithoutHeader", { uri: process.env.EXPO_PUBLIC_WEB_URL + "/zone/pay/" + data?.id + "/h5_zone_superfanship/0", }) } style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/chaofen.png")} /> <Text style={tailwind("text-white text-xs font-medium mt-0.5")} > {data?.is_superfanship_unlocked === 1 ? "尊贵超粉" : "成为超粉"} </Text> </TouchableOpacity> )} <TouchableOpacity onPress={() => setIsIronFanModalVisible(true)} style={tailwind("flex flex-col items-center")} > <NativeImage source={require("../../assets/icon/others/tiefen.png")} /> <View style={tailwind("flex flex-col items-center")}> <Text style={tailwind("text-white text-xs font-medium mt-0.5")} > {data?.is_ironfanship_unlocked === 1 ? "已是铁粉" : "成为铁粉"} </Text> <Text style={{ fontSize: 10, ...tailwind("text-[#FFFFFF80] font-medium mt-0.5"), }} > {`${parseInt(data?.expenditure / 100, 10)}/${parseInt( data?.ironfanship_price / 100, 10 )}`} </Text> </View> </TouchableOpacity> </View> )} {/* 点击成为铁粉Modal */} <BecomeIronFanModal /> {/* 创建帖子Modal */} <CreatePostModal visible={isCreatePostModalVisible} setVisible={setIsCreatePostModalVisible} /> {/* 查看微信Modal */} {data?.streamer_ext?.wechat_lock_type === 0 ? ( <GetWechatModal visible={isAddWechatModalVisible} setVisible={setIsAddWechatModalVisible} streamerMid={route.params.mid} /> ) : ( <SubmitWechatModal visible={isAddWechatModalVisible} setVisible={setIsAddWechatModalVisible} streamerMid={route.params.mid} /> )} {/* 退款中禁止查看Modal */} <MyModal visible={isRefundingModalVisible} setVisible={setIsRefundingModalVisible} title="当前空间正在退款中" content="退款中空间不支持查看,请关注原支付渠道退款进度,退款后无法再次进入当前空间。" cancel={() => { setIsRefundingModalVisible(false); setTimeout(() => navigation.replace("HomeTab"), 500); }} confirm={() => { setIsRefundingModalVisible(false); setTimeout(() => navigation.replace("HomeTab"), 500); }} /> </View> {/* 刷新按钮 */} <TouchableOpacity style={{ ...tailwind("absolute bg-[#13121F] p-1 rounded-full"), bottom: insets.bottom + 100, right: 16, }} onPress={startRotation} > <Animated.View style={{ transform: [{ rotate: spin }] }}> <Svg viewBox="0 0 1024 1024" width="30" height="30"> <Path d="M855.7 546.3c-16.6 0-30-13.5-30-30.1 0.1-72.8-25.1-143.9-71-200.4l-1-1.3c-31.5-38.5-72.1-69.2-117.3-88.9l-0.6-0.2-2.2-1c-6.8-2.9-14.3-5.6-22.9-8.4l-0.6-0.2-6.5-2.3c-7.1-2.1-14.7-3.9-20.6-5.3-1.2-0.3-2.3-0.5-3.5-0.8-2.1-0.5-4-1-5.6-1.3l-0.4-0.1c-6.8-1.4-14.1-2.5-21.7-3.3l-0.8-0.1-8.4-1.2c-73.7-7.1-147.4 12.6-207.8 55.6-13.5 9.6-32.2 6.5-41.8-7-9.6-13.5-6.5-32.2 7-41.8 72.3-51.5 160.8-75.1 249.1-66.4l1.3 0.2 8.7 1.2c9 1 17.8 2.4 26 4 3.1 0.6 6 1.3 8.6 1.9 1 0.2 1.9 0.5 2.9 0.7 7 1.6 16.2 3.8 25.2 6.5l1.2 0.4 6.9 2.4c10.2 3.4 19.3 6.7 27.7 10.3l0.8 0.4 2.4 1.1c54.1 23.6 102.4 60.4 139.9 106.3l1 1.4c54.4 67.1 84.2 151.5 84.1 237.9-0.1 16.4-13.6 29.8-30.1 29.8z" fill="#ffffff" ></Path> <Path d="M872.5 661.6l78.9-114.5c9-13.1-0.3-30.9-16.3-30.9H777.4c-15.9 0-25.3 17.8-16.3 30.9L840 661.6c7.9 11.4 24.7 11.4 32.5 0zM513.3 884.1c-12 0-24-0.6-36-1.8l-1.3-0.2-8.6-1.2c-9-1-17.8-2.4-26-4-3.1-0.6-6-1.3-8.6-1.9-1-0.2-1.9-0.5-2.9-0.7-7-1.6-16.2-3.8-25.2-6.5l-1.2-0.4-6.9-2.4c-10.2-3.4-19.3-6.7-27.7-10.3l-0.8-0.4-2.4-1.1c-54.1-23.6-102.4-60.4-139.9-106.3l-1-1.4c-54.4-67.1-84.2-151.5-84.1-237.9 0-16.6 13.5-29.9 30-29.9h0.1c16.6 0 30 13.5 29.9 30.1-0.1 72.8 25.1 143.9 71 200.4l1 1.3c31.5 38.5 72.1 69.2 117.3 88.9l2.8 1.3c6.8 2.9 14.3 5.6 22.9 8.4l0.6 0.2 6.5 2.3c7.1 2.1 14.7 3.9 20.6 5.3 1.2 0.3 2.3 0.5 3.5 0.8 2.1 0.5 4 1 5.6 1.3l0.4 0.1c6.8 1.3 14.1 2.5 21.7 3.3l0.8 0.1 8.4 1.2c73.7 7.1 147.4-12.6 207.8-55.6 13.5-9.6 32.2-6.5 41.8 7 9.6 13.5 6.5 32.2-7 41.8-62.5 44.5-137 68.2-213.1 68.2z" fill="#ffffff" ></Path> <Path d="M153.8 362.4L74.9 476.9c-9 13.1 0.3 30.9 16.3 30.9H249c15.9 0 25.3-17.8 16.3-30.9l-78.9-114.5c-7.9-11.4-24.7-11.4-32.6 0z" fill="#ffffff" ></Path> </Svg> </Animated.View> </TouchableOpacity> </BottomSheetModalProvider> </GestureHandlerRootView> ); }