tiefen_space_app/screeens/Search/index.jsx

1340 lines
42 KiB
React
Raw Normal View History

2024-08-21 17:54:04 +08:00
import {
View,
Text,
Image as NativeImage,
ScrollView,
TouchableOpacity,
2024-08-23 14:06:27 +08:00
Dimensions,
2024-08-23 19:03:05 +08:00
Platform,
2024-08-21 17:54:04 +08:00
} from "react-native";
2024-09-04 12:23:56 +08:00
import React, { useState, useEffect, useRef, useCallback } from "react";
2023-12-29 00:27:44 +08:00
import { useTailwind } from "tailwind-rn";
2024-08-21 17:54:04 +08:00
import {
useSafeAreaInsets,
SafeAreaProvider,
} from "react-native-safe-area-context";
import {
SearchBar,
ListItem,
Icon,
CheckBox,
BottomSheet,
} from "@rneui/themed";
2023-12-29 00:27:44 +08:00
import { Image } from "expo-image";
2024-09-04 12:23:56 +08:00
import { useFocusEffect } from "@react-navigation/native";
2023-12-29 00:27:44 +08:00
import Empty from "../../components/Empty";
import Toast from "react-native-toast-message";
import baseRequest from "../../utils/baseRequest";
import { generateSignature } from "../../utils/crypto";
2024-06-06 21:06:51 +08:00
import MyDivider from "../../components/MyDivider/index";
2024-08-16 23:55:29 +08:00
import MySlider from "../../components/MySlider";
2024-08-21 17:54:04 +08:00
import Picker from "../../components/Picker";
2024-08-23 14:06:27 +08:00
import { get } from "../../utils/storeInfo";
2024-09-04 18:11:20 +08:00
const filterComprehensiveItems = {
age: { lower_bound: 18, upper_bound: 60 },
fans: { lower_bound: 1, upper_bound: 1000 },
weight: { lower_bound: 35, upper_bound: 100 },
height: { lower_bound: 140, upper_bound: 200 },
city: "",
constellation: "",
is_active_within_a_week: 0,
comprehensiveUsed: { show: false, used: false },
};
const filterPriceItems = {
zone_admission_price: { lower_bound: 0, upper_bound: 4000 },
wechat_coin_price: { lower_bound: 0, upper_bound: 100000 },
priceUsed: { show: false, used: false },
};
2023-12-29 00:27:44 +08:00
export default function Search({ navigation, route }) {
const tailwind = useTailwind();
const insets = useSafeAreaInsets();
const searchRef = useRef(null);
2024-08-23 14:06:27 +08:00
const screenDimensions = Dimensions.get("screen");
2023-12-29 00:27:44 +08:00
const [search, setSearch] = useState("");
2024-08-23 14:06:27 +08:00
const [isMember, setIsMember] = useState(0);
2024-06-06 21:06:51 +08:00
const [streamers, setStreamers] = useState([]);
2024-08-21 17:54:04 +08:00
const [recommList, setRecommList] = useState([]);
2024-06-06 21:06:51 +08:00
const [zones, setZones] = useState([]);
2023-12-29 00:27:44 +08:00
const [isloading, setIsloading] = useState(false);
2024-08-23 21:38:23 +08:00
const [dontScroll, setDontScroll] = useState(true);
2024-08-21 17:54:04 +08:00
const [isFilterVisible, setIsFilterVisible] = useState({
comprehensive: false,
zone_admission_price: false,
});
const filters = [
{
name: "年龄",
upper_bound: 60,
lower_bound: 18,
2024-08-23 14:06:27 +08:00
default: 19,
2024-08-21 17:54:04 +08:00
step: 1,
type: "slider",
unit: "岁",
key: "age",
2024-08-23 14:06:27 +08:00
stepValues: Array(61)
.fill(null)
.map((_, index) => index),
2024-08-21 17:54:04 +08:00
},
{
name: "全网粉丝",
upper_bound: 200,
lower_bound: 1,
default: 24,
2024-09-09 15:13:44 +08:00
step: 200 / 9,
2024-08-21 17:54:04 +08:00
type: "slider",
unit: "万",
key: "fans",
2024-09-09 15:13:44 +08:00
stepValues: [1, 5, 10, 20, 50, 75, 100, 150, 200, 1000],
2024-08-21 17:54:04 +08:00
},
{
name: "身高",
upper_bound: 200,
lower_bound: 140,
default: 160,
step: 1,
type: "slider",
2024-08-30 13:55:31 +08:00
unit: "cm",
2024-08-21 17:54:04 +08:00
key: "height",
2024-08-23 14:06:27 +08:00
stepValues: Array(201)
.fill(null)
.map((_, index) => index),
2024-08-21 17:54:04 +08:00
},
{
name: "体重",
upper_bound: 100,
lower_bound: 35,
default: 50,
step: 1,
type: "slider",
2024-08-30 13:55:31 +08:00
unit: "kg",
2024-08-21 17:54:04 +08:00
key: "weight",
2024-08-23 14:06:27 +08:00
stepValues: Array(101)
.fill(null)
.map((_, index) => index),
2024-08-21 17:54:04 +08:00
},
{
name: "所在地",
upper_bound: 1000,
lower_bound: 10,
2024-08-30 13:55:31 +08:00
default: 0,
2024-08-21 17:54:04 +08:00
step: 1,
type: "select",
items: [
"北京市",
"天津市",
"河北省",
"山西省",
"内蒙古自治区",
"辽宁省",
"吉林省",
"黑龙江省",
"上海市",
"江苏省",
"浙江省",
"安徽省",
"福建省",
"江西省",
"山东省",
"河南省",
"湖北省",
"湖南省",
"广东省",
"广西壮族自治区",
"海南省",
"重庆市",
"四川省",
"贵州省",
"云南省",
"西藏自治区",
"陕西省",
"甘肃省",
"青海省",
"宁夏回族自治区",
"新疆维吾尔自治区",
"台湾省",
"香港特别行政区",
"澳门特别行政区",
],
key: "city",
},
{
name: "星座",
upper_bound: 1000,
lower_bound: 10,
2024-08-30 13:55:31 +08:00
default: 0,
2024-08-21 17:54:04 +08:00
step: 1,
type: "select",
key: "constellation",
items: [
"白羊座",
"‌金牛座",
"‌双子座",
"‌巨蟹座",
"‌狮子座",
"‌处女座",
"‌天秤座",
"‌天蝎座",
"‌射手座",
"‌摩羯座",
"‌水瓶座",
"‌双鱼座",
],
},
{
name: "最近7日内空间有更新",
upper_bound: 1000,
lower_bound: 10,
default: 24,
step: 1,
type: "checkbox",
key: "is_active_within_a_week",
},
];
const priceFilters = [
{
name: "空间价格",
2024-09-09 15:13:44 +08:00
upper_bound: 300,
2024-08-21 17:54:04 +08:00
lower_bound: 0,
default: 24,
2024-09-09 15:13:44 +08:00
step: 300 / 7,
2024-08-21 17:54:04 +08:00
type: "slider",
unit: "¥",
key: "zone",
2024-09-09 15:13:44 +08:00
stepValues: [0, 10, 50, 100, 150, 200, 300, 4000],
2024-08-21 17:54:04 +08:00
},
{
name: "微信价格",
2024-09-09 17:37:22 +08:00
desc: "10金币=1元",
2024-09-09 15:13:44 +08:00
upper_bound: 30000,
2024-08-21 17:54:04 +08:00
lower_bound: 0,
default: 24,
2024-09-09 15:13:44 +08:00
step: 30000 / 9,
2024-08-21 17:54:04 +08:00
type: "slider",
unit: "金币",
key: "wechat",
2024-09-09 15:13:44 +08:00
stepValues: [0, 100, 300, 500, 1000, 5000, 10000, 20000, 30000, 100000],
2024-08-21 17:54:04 +08:00
},
];
const [filtersValue, setFiltersValue] = useState({
2024-09-04 18:11:20 +08:00
...filterComprehensiveItems,
...filterPriceItems,
});
const [currentFiltersChangeValue, setCurrentFiltersChangeValue] = useState({
...filterComprehensiveItems,
...filterPriceItems,
2024-08-21 17:54:04 +08:00
});
2023-12-29 00:27:44 +08:00
const updateSearch = (search) => {
setSearch(search);
if (!search) return;
setIsloading(true);
};
//进入页面默认focus
useEffect(() => {
searchRef.current.focus();
}, []);
2024-09-04 12:23:56 +08:00
useFocusEffect(
useCallback(() => {
const getIsMember = async () => {
const apiUrl = process.env.EXPO_PUBLIC_API_URL;
let api;
api = "/api/account/list_by_mid";
const account = await get("account");
try {
const base = await baseRequest();
const signature = await generateSignature({
...base,
mid: account.mid,
});
const response = await fetch(
`${apiUrl}${api}?signature=${signature}`,
{
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
...base,
mid: account.mid,
}),
}
);
const data = await response.json();
setIsMember(data.data.account.is_a_member);
} catch (error) {
console.error(error);
}
};
getIsMember();
}, [])
);
2023-12-29 00:27:44 +08:00
//搜索框文本变化时进行搜索
useEffect(() => {
if (!search) {
2024-06-06 21:06:51 +08:00
setStreamers([]);
setZones([]);
2023-12-29 00:27:44 +08:00
return;
}
const isNumeric = (str) => {
return /^\d+$/.test(str);
};
const getResult = async () => {
2024-09-04 12:23:56 +08:00
if (filtersValue.comprehensiveUsed.used || filtersValue.priceUsed.used) {
2024-09-04 18:11:20 +08:00
handleResetFiltersValue();
handleResetFiltersSearchValue();
2024-09-04 12:23:56 +08:00
}
2023-12-29 00:27:44 +08:00
const apiUrl = process.env.EXPO_PUBLIC_API_URL;
const isSearchInt = isNumeric(search);
let api;
let querryParams;
if (isSearchInt) {
api = "/api/streamer/list_ext_fuzzily_by_user_id";
querryParams = { user_id: parseInt(search, 10) };
} else {
api = "/api/streamer/list_ext_fuzzily_by_name";
querryParams = { name: search };
}
try {
const base = await baseRequest();
const signature = await generateSignature({
...base,
...querryParams,
offset: 0,
limit: 20,
});
const response = await fetch(`${apiUrl}${api}?signature=${signature}`, {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
...base,
...querryParams,
offset: 0,
limit: 20,
}),
});
const data = await response.json();
if (data.ret === -1) {
Toast.show({
type: "error",
text1: data.msg,
topOffset: 60,
});
return;
}
if (!ignore) {
2024-06-06 21:06:51 +08:00
const zonesData = data.data.list.filter(
(item) => item.zones.length > 0
);
setStreamers(data.data.list);
setZones(zonesData);
2024-09-04 12:23:56 +08:00
setRecommList([]);
2023-12-29 00:27:44 +08:00
}
setIsloading(false);
} catch (error) {
console.error(error);
}
};
let ignore = false;
getResult();
return () => {
ignore = true;
};
}, [search]);
2024-08-30 13:55:31 +08:00
const getFiltersResult = async (obj) => {
2024-09-04 12:23:56 +08:00
if (search != "") {
setSearch("");
}
2024-09-04 18:11:20 +08:00
setFiltersValue((old) => {
return {
...old,
...obj,
};
2024-09-04 12:23:56 +08:00
});
2024-08-21 17:54:04 +08:00
const apiUrl = process.env.EXPO_PUBLIC_API_URL;
2024-09-04 12:23:56 +08:00
const newObj = obj || { ...filtersValue, ...currentFiltersChangeValue };
2024-08-21 17:54:04 +08:00
let api;
let querryParams;
api = "/api/streamer/filter";
2024-08-30 13:55:31 +08:00
let currentFilterValue = {
...newObj,
zone_admission_price: {
lower_bound: newObj.zone_admission_price.lower_bound * 100,
upper_bound: newObj.zone_admission_price.upper_bound * 100,
},
};
2024-08-21 17:54:04 +08:00
delete currentFilterValue.comprehensiveUsed;
delete currentFilterValue.priceUsed;
querryParams = currentFilterValue;
2024-09-04 12:23:56 +08:00
2024-08-21 17:54:04 +08:00
try {
const base = await baseRequest();
const signature = await generateSignature({
...base,
...querryParams,
2024-08-30 13:55:31 +08:00
// offset: 0,
// limit: 20,
2024-08-21 17:54:04 +08:00
});
2024-09-09 15:13:44 +08:00
2024-08-21 17:54:04 +08:00
const response = await fetch(`${apiUrl}${api}?signature=${signature}`, {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify({
...base,
...querryParams,
2024-08-30 13:55:31 +08:00
// offset: 0,
// limit: 20,
2024-08-21 17:54:04 +08:00
}),
});
const data = await response.json();
if (data.ret === -1) {
Toast.show({
type: "error",
text1: data.msg,
topOffset: 60,
});
return;
}
const zonesData = data.data.streamer_list.filter(
(item) => item.zones.length > 0
);
setStreamers(data.data.streamer_list);
setZones(zonesData);
setRecommList(data.data.recomm_list);
setIsloading(false);
} catch (error) {
console.error(error);
}
};
2024-09-04 18:11:20 +08:00
// 更新MySlider值
const handleChangePriceSliderValue = useCallback((value, item) => {
if (item.key == "zone") {
setCurrentFiltersChangeValue((old) => ({
...old,
zone_admission_price: value,
}));
} else if (item.key == "wechat") {
setCurrentFiltersChangeValue((old) => ({
...old,
wechat_coin_price: value,
}));
}
}, []);
const handleChangeComprehensiveSliderValue = useCallback((value, item) => {
2024-09-07 00:15:17 +08:00
setCurrentFiltersChangeValue((old) => {
let newCurrentFiltersChangeValue = {
2024-09-04 18:11:20 +08:00
...old,
2024-09-07 00:15:17 +08:00
};
newCurrentFiltersChangeValue[item.key] = value;
return { ...old, ...newCurrentFiltersChangeValue };
});
2024-09-04 18:11:20 +08:00
}, []);
2024-09-04 12:23:56 +08:00
// 重置重置筛选值
const handleResetFiltersValue = (type) => {
let obj = filtersValue;
if (type == "comprehensive") {
obj = {
...filtersValue,
2024-09-04 18:11:20 +08:00
...filterComprehensiveItems,
2024-09-04 12:23:56 +08:00
};
} else if (type == "zone_admission_price") {
obj = {
...filtersValue,
2024-09-04 18:11:20 +08:00
...filterPriceItems,
};
} else {
obj = {
...filterComprehensiveItems,
...filterPriceItems,
2024-09-04 12:23:56 +08:00
};
2024-08-23 14:06:27 +08:00
}
2024-09-04 12:23:56 +08:00
setCurrentFiltersChangeValue((old) => {
const newObj = {
...old,
...obj,
};
return newObj;
});
2024-08-23 14:06:27 +08:00
};
2024-09-04 12:23:56 +08:00
//重置筛选查询值
const handleResetFiltersSearchValue = (type) => {
let obj = filtersValue;
2024-08-23 14:06:27 +08:00
if (type == "comprehensive") {
obj = {
...filtersValue,
2024-09-04 18:11:20 +08:00
...filterComprehensiveItems,
2024-08-23 14:06:27 +08:00
};
2024-09-04 12:23:56 +08:00
} else if (type == "zone_admission_price") {
2024-08-23 14:06:27 +08:00
obj = {
...filtersValue,
2024-09-04 18:11:20 +08:00
...filterPriceItems,
2024-08-23 14:06:27 +08:00
};
2024-09-04 12:23:56 +08:00
} else {
obj = {
2024-09-04 18:11:20 +08:00
...filterComprehensiveItems,
...filterPriceItems,
2024-09-04 12:23:56 +08:00
};
2024-08-23 14:06:27 +08:00
}
2024-08-30 13:55:31 +08:00
setFiltersValue((old) => {
const newObj = {
...old,
...obj,
};
if (!obj.comprehensiveUsed.used && !obj.priceUsed.used) {
setRecommList([]);
setStreamers([]);
setZones([]);
} else {
getFiltersResult(newObj);
}
return newObj;
});
2024-08-21 17:54:04 +08:00
};
2024-06-06 21:06:51 +08:00
//空间组件
2024-08-30 13:55:31 +08:00
const ZoneItem = ({ item, onlyOne }) => {
2024-06-06 21:06:51 +08:00
return (
<ListItem
onPress={() =>
navigation.navigate("SpaceIntroduce", { mid: item?.mid })
}
containerStyle={tailwind("p-0 bg-[#07050A]")}
>
<View style={tailwind("flex flex-col flex-1")}>
<View style={tailwind("flex-row py-3")}>
<Image
style={tailwind("w-12 h-12 rounded-full")}
source={item?.avatar?.images[0]?.urls[0]}
contentFit="cover"
cachePolicy="disk"
/>
<View style={tailwind("ml-2 justify-around flex-1")}>
<View style={tailwind("flex flex-row items-center")}>
<Text
style={tailwind("text-base text-white font-medium")}
numberOfLines={1}
ellipsizeMode="tail"
>
{item?.name}
</Text>
<View
style={tailwind(
"flex-row items-center py-0.5 px-2 ml-1 bg-[#FFFFFF1A] rounded-full"
)}
>
<NativeImage
source={require("../../assets/icon/12DP/ID.png")}
/>
<Text
style={tailwind("text-white text-xs font-medium ml-0.5")}
>
{item?.user_id}
</Text>
</View>
</View>
<Text
style={tailwind("text-sm text-[#FFFFFF80]")}
numberOfLines={1}
ellipsizeMode="tail"
>
2024-08-30 13:55:31 +08:00
{item.zones[0]?.profile}
2024-06-06 21:06:51 +08:00
</Text>
</View>
</View>
2024-08-30 13:55:31 +08:00
{onlyOne && (
2024-06-06 21:06:51 +08:00
<View
style={tailwind("flex flex-col bg-[#FFFFFF1A] rounded-xl mb-2")}
>
<Text
style={tailwind(
"text-[#FFFFFF80] text-sm font-medium mt-2 ml-2"
)}
>
Ta的动态·
{item.zones[0].zone_moment_count > 99
? "99+"
: item.zones[0].zone_moment_count}
{item.zones[0].image_count > 0 &&
` | 图片·${
item.zones[0].image_count > 99
? "99+"
: item.zones[0].image_count
}`}
{item.zones[0].video_count > 0 &&
` | 视频·${
item.zones[0].video_count > 99
? "99+"
: item.zones[0].video_count
}`}
</Text>
<View
style={{
aspectRatio: "1/1",
...tailwind("flex flex-row"),
}}
>
<View
style={{
flexBasis: "66.66%",
paddingTop: 8,
paddingRight: 8,
}}
>
<Image
style={tailwind("rounded-xl w-full h-full")}
source={item?.shorts?.videos[0]?.cover_urls[0]}
contentFit="cover"
cachePolicy="disk"
/>
<View
style={tailwind(
"flex absolute w-full h-full left-0 top-0 z-10 justify-center items-center"
)}
>
<NativeImage
source={require("../../assets/icon/others/play.png")}
/>
</View>
</View>
<View style={tailwind("flex flex-col basis-1/3")}>
<View
style={{
aspectRatio: "1/1",
paddingRight: 0,
paddingTop: 8,
}}
>
<Image
style={tailwind("rounded-xl w-full h-full")}
source={item?.cover?.images[0]?.urls[0]}
contentFit="cover"
cachePolicy="disk"
/>
</View>
{item.album.images.map((_item, index) => {
if (index > 1) return;
return (
<View
key={index}
style={{
aspectRatio: "1/1",
paddingRight: 0,
paddingTop: 8,
}}
>
<Image
style={tailwind("rounded-xl w-full h-full")}
source={_item?.urls[0]}
contentFit="cover"
cachePolicy="disk"
/>
</View>
);
})}
</View>
</View>
</View>
)}
</View>
</ListItem>
);
};
//主播组件
const StreamerItem = ({ item }) => {
2023-12-29 00:27:44 +08:00
return (
<ListItem
onPress={() =>
navigation.navigate("StreamerProfile", {
mid: item.mid,
})
}
containerStyle={tailwind("p-0 bg-[#07050A]")}
>
<View style={tailwind("flex-1")}>
<View style={tailwind("flex-row py-3")}>
<Image
style={tailwind("w-12 h-12 rounded-full")}
source={item?.avatar?.images[0]?.urls[0]}
contentFit="cover"
transition={1000}
cachePolicy="disk"
/>
<View style={tailwind("ml-2 justify-around flex-1")}>
<View style={tailwind("flex flex-row items-center")}>
<Text
style={tailwind("text-base text-white font-medium")}
numberOfLines={1}
ellipsizeMode="tail"
>
{item?.name}
</Text>
<View
style={tailwind(
"flex-row items-center py-0.5 px-2 ml-1 bg-[#FFFFFF1A] rounded-full"
)}
>
<NativeImage
source={require("../../assets/icon/12DP/ID.png")}
/>
<Text
style={tailwind("text-white text-xs font-medium ml-0.5")}
>
{item?.user_id}
</Text>
</View>
</View>
<Text
style={tailwind("text-sm text-[#FFFFFF80]")}
numberOfLines={1}
ellipsizeMode="tail"
>
{item.bio}
</Text>
</View>
</View>
</View>
</ListItem>
);
};
return (
<View
style={{
paddingTop: insets.top,
paddingBottom: insets.bottom,
paddingLeft: insets.left,
paddingRight: insets.right,
...tailwind("flex-1"),
}}
>
<View style={tailwind("flex-row px-2 items-center")}>
<Icon
type="ionicon"
name="chevron-back"
size={32}
color="white"
onPress={() => navigation.goBack()}
/>
<SearchBar
ref={searchRef}
containerStyle={tailwind("flex-1 bg-[#07050A]")}
inputContainerStyle={tailwind("h-10 bg-[#FFFFFF1A]")}
inputStyle={tailwind("text-white")}
placeholder="搜索Ta的昵称或id"
platform="ios"
cancelButtonProps={tailwind("text-[#FF669E]")}
cancelButtonTitle="清空"
clearIcon={() => <></>}
searchIcon={() => <></>}
showLoading={isloading}
onChangeText={updateSearch}
value={search}
/>
</View>
2024-08-21 17:54:04 +08:00
<ScrollView style={tailwind("flex-1")}>
2024-08-23 19:03:05 +08:00
<View
style={{
2024-09-04 12:23:56 +08:00
...tailwind("flex flex-row px-4"),
2024-08-23 19:03:05 +08:00
justifyContent: "space-evenly",
}}
>
2024-08-21 17:54:04 +08:00
<TouchableOpacity
onPress={() => {
setIsFilterVisible({
...isFilterVisible,
comprehensive: !isFilterVisible.comprehensive,
});
}}
style={{
2024-09-04 12:23:56 +08:00
...tailwind(
"pl-4 mr-4 flex-1 rounded-xl flex-row items-center justify-around"
),
2024-08-21 17:54:04 +08:00
borderColor: "#ff75c8",
2024-08-23 14:06:27 +08:00
borderWidth: filtersValue.comprehensiveUsed.used ? 1 : 0,
2024-08-21 17:54:04 +08:00
boxSizing: "border-box",
2024-09-04 12:23:56 +08:00
// width: 110,
// width: "50%",
2024-08-21 17:54:04 +08:00
}}
>
<Text
style={{
...tailwind("text-white text-lg font-medium"),
marginVertical: 6,
2024-08-23 14:06:27 +08:00
color: filtersValue.comprehensiveUsed.used
? "#ff75c8"
: "#ffffff80",
2024-08-21 17:54:04 +08:00
}}
>
综合筛选
</Text>
2024-08-23 14:06:27 +08:00
{filtersValue.comprehensiveUsed.used && (
2024-08-21 17:54:04 +08:00
<TouchableOpacity
2024-09-04 18:11:20 +08:00
onPress={() => {
handleResetFiltersValue("comprehensive");
handleResetFiltersSearchValue("comprehensive");
}}
2024-09-04 12:23:56 +08:00
style={{
2024-09-09 13:12:18 +08:00
paddingHorizontal: 10,
2024-09-04 12:23:56 +08:00
height: 32,
display: "flex",
2024-09-09 13:12:18 +08:00
alignItems: "center",
2024-09-04 12:23:56 +08:00
justifyContent: "center",
backgroundColor: "#301024",
borderRadius: 12,
}}
2024-08-21 17:54:04 +08:00
>
<Icon
type="ionicon"
name="close"
size={18}
color="#ff75c8"
style={{ marginLeft: 2 }}
/>
</TouchableOpacity>
)}
</TouchableOpacity>
<TouchableOpacity
onPress={() => {
setIsFilterVisible({
...isFilterVisible,
zone_admission_price: !isFilterVisible.zone_admission_price,
});
}}
style={{
2024-09-04 12:23:56 +08:00
...tailwind(
"pl-4 rounded-xl flex-1 flex-row items-center justify-around"
),
2024-08-21 17:54:04 +08:00
borderColor: "#ff75c8",
2024-08-23 14:06:27 +08:00
borderWidth: filtersValue.priceUsed.used ? 1 : 0,
2024-08-21 17:54:04 +08:00
boxSizing: "border-box",
2024-09-04 12:23:56 +08:00
// width: 110,
width: "50%",
2024-08-21 17:54:04 +08:00
}}
>
<Text
style={{
...tailwind("text-white text-lg font-medium"),
2024-09-04 12:23:56 +08:00
paddingVertical: 6,
2024-08-23 14:06:27 +08:00
color: filtersValue.priceUsed.used ? "#ff75c8" : "#ffffff80",
2024-08-21 17:54:04 +08:00
}}
>
价格筛选
</Text>
2024-08-23 14:06:27 +08:00
{filtersValue.priceUsed.used && (
2024-08-21 17:54:04 +08:00
<TouchableOpacity
2024-09-04 18:11:20 +08:00
onPress={() => {
handleResetFiltersValue("zone_admission_price");
handleResetFiltersSearchValue("zone_admission_price");
}}
2024-09-04 12:23:56 +08:00
style={{
2024-09-09 13:12:18 +08:00
paddingHorizontal: 10,
2024-09-04 12:23:56 +08:00
height: 32,
display: "flex",
2024-09-09 13:12:18 +08:00
alignItems: "center",
2024-09-04 12:23:56 +08:00
justifyContent: "center",
backgroundColor: "#301024",
borderRadius: 12,
}}
2024-08-21 17:54:04 +08:00
>
<Icon
type="ionicon"
name="close"
size={18}
color="#ff75c8"
style={{ marginLeft: 2 }}
/>
</TouchableOpacity>
)}
</TouchableOpacity>
</View>
2024-08-23 14:06:27 +08:00
<View
style={{
display: "flex",
flex: 1,
}}
>
2024-08-21 17:54:04 +08:00
<BottomSheet
2024-08-23 14:06:27 +08:00
// containerStyle={{ marginTop: 80 }}
2024-08-23 19:03:05 +08:00
onBackdropPress={() =>
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
})
}
2024-08-23 14:06:27 +08:00
backdropStyle={{ backgroundColor: "#0000004d" }}
scrollViewProps={{
2024-08-23 21:38:23 +08:00
scrollEnabled: dontScroll,
2024-08-23 14:06:27 +08:00
contentContainerStyle: {
// paddingTop: 20,
backgroundColor: "black",
borderRadius: 12,
},
onScroll: (event) => {
const { y } = event.nativeEvent.contentOffset;
if (y < -50) {
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
});
}
},
}}
2024-08-21 17:54:04 +08:00
isVisible={isFilterVisible.comprehensive}
style={{
2024-08-23 19:03:05 +08:00
paddingTop: Platform.OS != "ios" ? 55 : 110,
2024-08-21 17:54:04 +08:00
}}
>
2024-08-23 14:06:27 +08:00
<View>
2024-08-21 17:54:04 +08:00
{filters.map((item, index) => (
<View
2024-08-23 19:03:05 +08:00
key={index}
2024-08-21 17:54:04 +08:00
style={{
2024-08-23 20:52:07 +08:00
...tailwind(`p-4 rounded-xl `),
2024-08-21 17:54:04 +08:00
backgroundColor: "#13121f",
margin: 18,
2024-08-23 20:52:07 +08:00
marginBottom: 3,
2024-08-21 17:54:04 +08:00
flexDirection: item.type !== "slider" ? "row" : "col",
justifyContent: "space-between",
2024-08-23 20:52:07 +08:00
alignItems: item.type == "slider" ? "flex-start" : "center",
2024-08-21 17:54:04 +08:00
// paddingBottom: 50,
}}
>
<Text
style={{
...tailwind("text-white font-medium"),
2024-08-23 20:52:07 +08:00
// marginVertical: 6,
2024-08-21 17:54:04 +08:00
}}
>
{item.name}
</Text>
{item.type == "slider" ? (
2024-08-23 20:52:07 +08:00
<View style={{ paddingVertical: 6 }}>
<MySlider
height={40}
lower_bound={item.lower_bound}
upper_bound={item.upper_bound}
itemKey={item.key}
2024-08-23 21:38:23 +08:00
handleFunc={() => {
setDontScroll((old) => !old);
}}
2024-08-23 20:52:07 +08:00
leftValue={
2024-09-04 18:11:20 +08:00
currentFiltersChangeValue[item.key]?.lower_bound
2024-08-21 17:54:04 +08:00
}
2024-08-23 20:52:07 +08:00
rightValue={
2024-09-04 18:11:20 +08:00
currentFiltersChangeValue[item.key]?.upper_bound
2024-08-23 20:52:07 +08:00
}
step={item.step}
hasInfinity={item.key == "fans"}
stepValues={item.stepValues}
2024-09-04 18:11:20 +08:00
onChange={(value) =>
handleChangeComprehensiveSliderValue(value, item)
}
2024-08-23 20:52:07 +08:00
maximumTrackTintColor="#382435"
minimumTrackTintColor="#ff75c8"
processHeight={5}
unit={item.unit}
2024-08-30 13:55:31 +08:00
unitSite={item.key == "zone" ? "left" : "right"}
2024-08-23 20:52:07 +08:00
thumbImage={require("../../assets/icon/32DP/edit.png")}
/>
</View>
2024-08-21 17:54:04 +08:00
) : item.type == "checkbox" ? (
<CheckBox
2024-09-04 12:23:56 +08:00
checked={
currentFiltersChangeValue?.is_active_within_a_week
}
onPress={
() =>
2024-09-07 00:15:17 +08:00
// setCurrentFiltersChangeValue((old) => ({
// ...old,
// is_active_within_a_week: old.is_active_within_a_week
// ? 0
// : 1,
// }))
handleChangeComprehensiveSliderValue(
currentFiltersChangeValue.is_active_within_a_week
2024-09-04 12:23:56 +08:00
? 0
: 1,
2024-09-07 00:15:17 +08:00
item
)
2024-09-04 12:23:56 +08:00
// setFiltersValue((old) => ({
// ...old,
// is_active_within_a_week: old.is_active_within_a_week
// ? 0
// : 1,
// }))
2024-08-21 17:54:04 +08:00
}
iconType="material-community"
checkedIcon="checkbox-marked"
uncheckedIcon="checkbox-blank-outline"
checkedColor="#FF669E"
containerStyle={tailwind("p-0 m-0 bg-transparent")}
size={24}
/>
) : (
<Picker
items={item.items?.map((it, index) => ({
label: it,
value: it,
}))}
2024-08-30 13:55:31 +08:00
placeholder={{ label: "不限", value: "" }}
2024-08-21 17:54:04 +08:00
value={
item.key == "constellation"
2024-09-04 12:23:56 +08:00
? currentFiltersChangeValue?.constellation
: currentFiltersChangeValue?.city
2024-08-21 17:54:04 +08:00
}
onChange={(value) =>
2024-09-07 00:15:17 +08:00
handleChangeComprehensiveSliderValue(value, item)
2024-08-21 17:54:04 +08:00
}
/>
)}
</View>
))}
2024-08-23 14:06:27 +08:00
<View
style={{
2024-08-23 19:03:05 +08:00
...tailwind("flex-row justify-around items-center mb-4 p-4"),
2024-08-23 14:06:27 +08:00
}}
>
2024-08-23 19:03:05 +08:00
{Platform.OS != "ios" && (
<View
style={{
...tailwind(
"p-2 rounded-xl flex justify-center items-center mr-2"
),
backgroundColor: "#ffffff1a",
width: 46,
height: 46,
}}
>
<Icon
type="ionicon"
name="close"
size={32}
color="white"
2024-09-04 12:23:56 +08:00
onPress={() => {
2024-08-23 19:03:05 +08:00
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
2024-09-04 12:23:56 +08:00
});
setCurrentFiltersChangeValue(filtersValue);
}}
2024-08-23 19:03:05 +08:00
/>
</View>
)}
2024-08-23 14:06:27 +08:00
{isMember == 1 && (
<TouchableOpacity
2024-08-21 17:54:04 +08:00
style={{
2024-08-23 20:52:07 +08:00
...tailwind("text-white px-4 py-2 rounded-xl mr-2"),
2024-08-23 14:06:27 +08:00
marginVertical: 6,
backgroundColor: "#ffffff1a",
width: 120,
2024-08-21 17:54:04 +08:00
}}
2024-08-23 14:06:27 +08:00
onPress={() => handleResetFiltersValue("comprehensive")}
2024-08-21 17:54:04 +08:00
>
2024-08-23 14:06:27 +08:00
<Text
style={{
...tailwind(
"text-white text-lg font-medium text-center"
),
}}
>
重置
</Text>
</TouchableOpacity>
)}
2024-08-21 17:54:04 +08:00
<TouchableOpacity
onPress={() => {
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
});
2024-08-23 14:06:27 +08:00
if (!isMember) {
navigation.navigate("WebWithoutHeader", {
uri: process.env.EXPO_PUBLIC_WEB_URL + "/vip",
});
return;
}
2024-09-04 12:23:56 +08:00
2024-09-04 18:11:20 +08:00
getFiltersResult({
...currentFiltersChangeValue,
priceUsed: filtersValue.priceUsed,
});
2024-08-23 14:06:27 +08:00
setFiltersValue((old) => ({
...old,
comprehensiveUsed: { show: false, used: true },
}));
2024-08-21 17:54:04 +08:00
}}
style={{
...tailwind("text-white px-4 py-2 rounded-xl"),
marginVertical: 6,
backgroundColor: "#ff75c8",
2024-08-23 14:06:27 +08:00
minWidth: 120,
2024-08-23 19:03:05 +08:00
flex: 1,
2024-08-21 17:54:04 +08:00
}}
>
<Text
style={{
...tailwind(
"text-white text-lg font-medium text-center"
),
}}
>
2024-08-23 14:06:27 +08:00
{!isMember ? "开通VIP后即可筛选" : "确定"}
2024-08-21 17:54:04 +08:00
</Text>
</TouchableOpacity>
</View>
</View>
</BottomSheet>
<BottomSheet
2024-08-23 14:06:27 +08:00
backdropStyle={{ backgroundColor: "#0000004d" }}
2024-08-23 19:03:05 +08:00
onBackdropPress={() =>
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
})
}
2024-08-23 14:06:27 +08:00
scrollViewProps={{
2024-08-23 21:38:23 +08:00
scrollEnabled: dontScroll,
2024-08-23 14:06:27 +08:00
contentContainerStyle: {
// paddingTop: 20,
backgroundColor: "black",
borderRadius: 12,
},
onScroll: (event) => {
const { y } = event.nativeEvent.contentOffset;
if (y < -50) {
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
});
}
},
}}
2024-08-21 17:54:04 +08:00
isVisible={isFilterVisible.zone_admission_price}
style={{
2024-08-23 19:03:05 +08:00
paddingTop: Platform.OS != "ios" ? 55 : 110,
2024-08-21 17:54:04 +08:00
}}
>
<View
style={{
backgroundColor: "black",
borderRadius: 12,
2024-08-23 14:06:27 +08:00
// flex: 1,
// position: "relative",
2024-08-30 13:55:31 +08:00
height:
screenDimensions.height - (Platform.OS != "ios" ? 84 : 120),
2024-08-21 17:54:04 +08:00
}}
>
2024-08-23 19:03:05 +08:00
{priceFilters.map((item, index) => (
2024-08-21 17:54:04 +08:00
<View
2024-08-23 19:03:05 +08:00
key={index}
2024-08-21 17:54:04 +08:00
style={{
...tailwind(`p-4 rounded-xl`),
backgroundColor: "#13121f",
margin: 18,
marginBottom: 6,
flexDirection: item.type !== "slider" ? "row" : "col",
justifyContent: "space-between",
// alignItems: "center",
// paddingBottom: 50,
}}
>
<Text
style={{
...tailwind("text-white font-medium"),
marginVertical: 6,
}}
>
{item.name}
</Text>
<MySlider
height={40}
lower_bound={item.lower_bound}
upper_bound={item.upper_bound}
2024-08-23 14:06:27 +08:00
leftValue={
item.key == "zone"
2024-09-04 12:23:56 +08:00
? currentFiltersChangeValue?.zone_admission_price
?.lower_bound
: currentFiltersChangeValue?.wechat_coin_price
?.lower_bound
2024-08-23 14:06:27 +08:00
}
rightValue={
item.key == "zone"
2024-09-04 12:23:56 +08:00
? currentFiltersChangeValue?.zone_admission_price
?.upper_bound
: currentFiltersChangeValue?.wechat_coin_price
?.upper_bound
2024-08-23 14:06:27 +08:00
}
stepValues={item.stepValues}
2024-08-21 17:54:04 +08:00
step={item.step}
2024-08-23 21:38:23 +08:00
handleFunc={() => {
setDontScroll((old) => !old);
}}
2024-08-23 14:06:27 +08:00
hasInfinity={true}
itemKey={item.key}
2024-09-04 18:11:20 +08:00
onChange={(value) =>
handleChangePriceSliderValue(value, item)
}
2024-08-23 19:03:05 +08:00
maximumTrackTintColor="#382435"
2024-08-21 17:54:04 +08:00
minimumTrackTintColor="#ff75c8"
processHeight={5}
unit={item.unit}
2024-08-30 13:55:31 +08:00
unitSite={item.key == "zone" ? "left" : "right"}
2024-08-21 17:54:04 +08:00
thumbImage={require("../../assets/icon/32DP/edit.png")}
/>
</View>
))}
2024-09-09 15:13:44 +08:00
<Text
style={{
...tailwind("text-sm font-medium mt-1 px-4"),
color: "#ffffff80",
}}
>
*{priceFilters[1]?.desc}
</Text>
2024-08-23 14:06:27 +08:00
<View
style={{
2024-08-30 13:55:31 +08:00
...tailwind("flex-row justify-around items-center mb-4 p-4"),
2024-08-23 14:06:27 +08:00
position: "absolute",
bottom: 0,
width: "100%",
}}
>
2024-08-23 19:03:05 +08:00
{Platform.OS != "ios" && (
<View
style={{
...tailwind(
"p-2 rounded-xl flex justify-center items-center mr-2"
),
backgroundColor: "#ffffff1a",
width: 46,
height: 46,
}}
>
<Icon
type="ionicon"
name="close"
size={32}
color="white"
2024-09-04 12:23:56 +08:00
onPress={() => {
2024-08-23 19:03:05 +08:00
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
2024-09-04 12:23:56 +08:00
});
setCurrentFiltersChangeValue(filtersValue);
}}
2024-08-23 19:03:05 +08:00
/>
</View>
)}
2024-08-23 14:06:27 +08:00
{isMember == 1 && (
<TouchableOpacity
2024-08-21 17:54:04 +08:00
style={{
2024-08-23 20:52:07 +08:00
...tailwind("text-white px-4 py-2 rounded-xl mr-2"),
2024-08-23 14:06:27 +08:00
marginVertical: 6,
backgroundColor: "#ffffff1a",
width: 120,
2024-08-21 17:54:04 +08:00
}}
2024-08-23 14:06:27 +08:00
onPress={() =>
handleResetFiltersValue("zone_admission_price")
}
2024-08-21 17:54:04 +08:00
>
2024-08-23 14:06:27 +08:00
<Text
style={{
...tailwind(
"text-white text-lg font-medium text-center"
),
}}
>
重置
</Text>
</TouchableOpacity>
)}
2024-08-21 17:54:04 +08:00
<TouchableOpacity
onPress={() => {
setIsFilterVisible({
zone_admission_price: false,
comprehensive: false,
});
2024-08-23 14:06:27 +08:00
if (!isMember) {
navigation.navigate("WebWithoutHeader", {
uri: process.env.EXPO_PUBLIC_WEB_URL + "/vip",
});
return;
}
2024-09-04 18:11:20 +08:00
getFiltersResult({
...currentFiltersChangeValue,
comprehensiveUsed: filtersValue.comprehensiveUsed,
});
2024-08-23 14:06:27 +08:00
setFiltersValue((old) => ({
...old,
priceUsed: { show: false, used: true },
}));
2024-08-21 17:54:04 +08:00
}}
style={{
...tailwind("text-white px-4 py-2 rounded-xl"),
2024-08-30 13:55:31 +08:00
marginVertical: 6,
2024-08-21 17:54:04 +08:00
backgroundColor: "#ff75c8",
2024-08-23 14:06:27 +08:00
minWidth: 120,
2024-08-23 19:03:05 +08:00
flex: 1,
2024-08-21 17:54:04 +08:00
}}
>
<Text
style={{
...tailwind(
"text-white text-lg font-medium text-center"
),
}}
>
2024-08-23 14:06:27 +08:00
{!isMember ? "开通VIP后即可筛选" : "确定"}
2024-08-21 17:54:04 +08:00
</Text>
</TouchableOpacity>
</View>
</View>
</BottomSheet>
2024-08-23 14:06:27 +08:00
</View>
2024-08-21 17:54:04 +08:00
<View style={tailwind("flex-1 px-4")}>
{zones.length > 0 && (
<Text style={tailwind("mt-2 text-white text-xl font-medium")}>
空间
</Text>
)}
{zones?.map((item, index) => (
2024-08-30 13:55:31 +08:00
<ZoneItem key={index} item={item} onlyOne={zones.length == 1} />
2024-08-21 17:54:04 +08:00
))}
{zones.length > 0 && <MyDivider style={tailwind("my-2")} />}
{streamers.length > 0 && (
<Text style={tailwind("mt-2 text-white text-xl font-medium")}>
用户
</Text>
)}
{streamers?.map((item, index) => (
<StreamerItem key={index} item={item} />
))}
2024-09-04 12:23:56 +08:00
{recommList.length > 0 && streamers.length > 0 && (
<MyDivider style={tailwind("my-2")} />
)}
2024-08-21 17:54:04 +08:00
{recommList.length > 0 && (
<Text style={tailwind("mt-2 text-white text-xl font-medium")}>
猜你喜欢
</Text>
)}
2024-08-30 13:55:31 +08:00
{recommList
.filter((it) => it.zones.length > 0)
?.map((item, index) => (
<ZoneItem key={index} item={item} onlyOne={false} />
))}
2024-08-21 17:54:04 +08:00
{streamers.length === 0 && recommList.length === 0 && (
<Empty type={search ? "search" : "nodata"} />
)}
</View>
2024-06-06 21:06:51 +08:00
</ScrollView>
2023-12-29 00:27:44 +08:00
</View>
);
}