2024-03-15 20:14:22 +08:00
|
|
|
|
import {
|
|
|
|
|
View,
|
|
|
|
|
Text,
|
|
|
|
|
Modal,
|
|
|
|
|
TouchableOpacity,
|
|
|
|
|
ActivityIndicator,
|
2024-03-23 15:28:01 +08:00
|
|
|
|
Platform,
|
2024-03-15 20:14:22 +08:00
|
|
|
|
} from "react-native";
|
|
|
|
|
import React, { useState, useCallback, useEffect } from "react";
|
|
|
|
|
import { ImageViewer as OriginImageViewer } from "react-native-image-zoom-viewer";
|
|
|
|
|
import { Icon } from "@rneui/themed";
|
|
|
|
|
import { useTailwind } from "tailwind-rn";
|
|
|
|
|
import { useSafeAreaInsets } from "react-native-safe-area-context";
|
|
|
|
|
import saveImage from "../../utils/saveImage";
|
|
|
|
|
import Toast from "react-native-toast-message";
|
|
|
|
|
import { get } from "../../utils/storeInfo";
|
|
|
|
|
import { useNavigation } from "@react-navigation/native";
|
|
|
|
|
import MyModal from "../MyModal";
|
2024-04-23 22:22:18 +08:00
|
|
|
|
import { Image } from "expo-image";
|
2024-04-24 16:33:47 +08:00
|
|
|
|
import { Button } from "@rneui/themed";
|
2024-04-23 22:22:18 +08:00
|
|
|
|
import { LinearGradient } from "expo-linear-gradient";
|
2024-12-20 20:40:46 +08:00
|
|
|
|
import { generateSignature } from "../../utils/crypto";
|
|
|
|
|
import baseRequest from "../../utils/baseRequest";
|
2024-03-15 20:14:22 +08:00
|
|
|
|
|
2024-12-20 20:40:46 +08:00
|
|
|
|
//imageUrls [{ url: string , id: int }] 网络图片需要查看原图时才传id,其他情况不用
|
2024-03-15 20:14:22 +08:00
|
|
|
|
//index int
|
2024-04-24 16:33:47 +08:00
|
|
|
|
//lockedStartIndex int
|
|
|
|
|
//onPressUnlockBtn () => void
|
2024-03-15 20:14:22 +08:00
|
|
|
|
|
2024-04-23 22:22:18 +08:00
|
|
|
|
export function ImageViewer({
|
|
|
|
|
isShow,
|
|
|
|
|
onClose,
|
2024-04-24 16:33:47 +08:00
|
|
|
|
onChange,
|
2024-03-15 20:14:22 +08:00
|
|
|
|
imageUrls,
|
|
|
|
|
index,
|
2024-04-23 22:22:18 +08:00
|
|
|
|
lockedStartIndex,
|
2024-04-24 16:33:47 +08:00
|
|
|
|
onPressUnlockBtn,
|
2024-12-20 20:40:46 +08:00
|
|
|
|
setImageViewerProps,
|
2024-03-15 20:14:22 +08:00
|
|
|
|
}) {
|
|
|
|
|
const tailwind = useTailwind();
|
|
|
|
|
const insets = useSafeAreaInsets();
|
|
|
|
|
|
|
|
|
|
const navigation = useNavigation();
|
|
|
|
|
|
2024-04-23 22:22:18 +08:00
|
|
|
|
const [isVisible, setIsVisible] = useState(false);
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
setIsVisible(isShow);
|
|
|
|
|
}, [isShow]);
|
|
|
|
|
|
|
|
|
|
const closeImageViewer = () => {
|
|
|
|
|
setIsVisible(false);
|
|
|
|
|
onClose && onClose();
|
|
|
|
|
};
|
|
|
|
|
|
2024-03-23 15:28:01 +08:00
|
|
|
|
const checkRole = useCallback(async () => {
|
|
|
|
|
const account = await get("account");
|
|
|
|
|
const role = account.role;
|
|
|
|
|
const isVip = account.is_a_member;
|
|
|
|
|
if (role !== 0 || isVip === 1) return true;
|
|
|
|
|
return false;
|
2024-03-15 20:14:22 +08:00
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
const [isVipModalVisible, setIsVipModalVisible] = useState(false);
|
|
|
|
|
const [isSaving, setIsSaving] = useState(false);
|
|
|
|
|
const hanldSaveImage = async (index) => {
|
2024-03-23 15:28:01 +08:00
|
|
|
|
const isVip = await checkRole();
|
2024-03-15 20:14:22 +08:00
|
|
|
|
if (!isVip) {
|
|
|
|
|
setIsVipModalVisible(true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (isSaving) return;
|
|
|
|
|
setIsSaving(true);
|
2024-04-24 16:33:47 +08:00
|
|
|
|
await saveImage(imageUrls[index].url);
|
2024-03-15 20:14:22 +08:00
|
|
|
|
setIsSaving(false);
|
|
|
|
|
};
|
|
|
|
|
|
2024-12-20 20:40:46 +08:00
|
|
|
|
//查看原图
|
|
|
|
|
const handleCheckOriginalImage = async (index) => {
|
|
|
|
|
const isVip = await checkRole();
|
|
|
|
|
if (!isVip) {
|
|
|
|
|
setIsVipModalVisible(true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
const apiUrl = process.env.EXPO_PUBLIC_API_URL;
|
|
|
|
|
try {
|
|
|
|
|
const base = await baseRequest();
|
|
|
|
|
const body = {
|
|
|
|
|
image_id: imageUrls[index].id,
|
|
|
|
|
...base,
|
|
|
|
|
};
|
|
|
|
|
const signature = await generateSignature(body);
|
|
|
|
|
const _response = await fetch(
|
|
|
|
|
`${apiUrl}/api/previews/original_image?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;
|
|
|
|
|
}
|
|
|
|
|
setImageViewerProps((prev) => {
|
|
|
|
|
let newImageViewerProps = prev;
|
|
|
|
|
newImageViewerProps.imageUrls[index] = {
|
|
|
|
|
id: imageUrls[index].id,
|
|
|
|
|
url: _data?.data?.urls[0],
|
|
|
|
|
};
|
|
|
|
|
return { ...newImageViewerProps };
|
|
|
|
|
});
|
|
|
|
|
Toast.show({
|
|
|
|
|
type: "success",
|
|
|
|
|
text1: "已切换至原图",
|
|
|
|
|
topOffset: 60,
|
|
|
|
|
});
|
|
|
|
|
} catch (error) {
|
|
|
|
|
console.error(error);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2024-03-15 20:14:22 +08:00
|
|
|
|
return (
|
|
|
|
|
<Modal visible={isVisible} statusBarTranslucent transparent={true}>
|
|
|
|
|
<OriginImageViewer
|
|
|
|
|
imageUrls={imageUrls}
|
|
|
|
|
index={index}
|
2024-04-23 22:22:18 +08:00
|
|
|
|
onClick={closeImageViewer}
|
|
|
|
|
onSwipeDown={closeImageViewer}
|
2024-04-24 16:33:47 +08:00
|
|
|
|
onChange={onChange ? (index) => onChange(index) : () => null}
|
2024-03-15 20:14:22 +08:00
|
|
|
|
enableSwipeDown
|
|
|
|
|
backgroundColor="#07050A"
|
2024-04-23 22:22:18 +08:00
|
|
|
|
renderImage={(props) => {
|
|
|
|
|
if (index >= lockedStartIndex) {
|
|
|
|
|
return (
|
|
|
|
|
<View>
|
|
|
|
|
<Image
|
|
|
|
|
{...props}
|
|
|
|
|
blurRadius={Platform.OS === "ios" ? 100 : 10}
|
|
|
|
|
/>
|
|
|
|
|
<View
|
|
|
|
|
style={tailwind(
|
|
|
|
|
"absolute flex justify-center items-center top-0 left-0 bg-[#00000080] w-full h-full"
|
|
|
|
|
)}
|
|
|
|
|
>
|
|
|
|
|
<Button
|
|
|
|
|
onPress={() => {
|
|
|
|
|
closeImageViewer();
|
2024-04-24 16:33:47 +08:00
|
|
|
|
onPressUnlockBtn();
|
2024-04-23 22:22:18 +08:00
|
|
|
|
}}
|
|
|
|
|
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")}
|
2024-04-24 16:33:47 +08:00
|
|
|
|
containerStyle={tailwind("w-3/4 px-4")}
|
2024-04-23 22:22:18 +08:00
|
|
|
|
>
|
|
|
|
|
立即解锁
|
|
|
|
|
</Button>
|
|
|
|
|
</View>
|
|
|
|
|
</View>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
return <Image {...props} />;
|
|
|
|
|
}}
|
2024-05-27 13:21:36 +08:00
|
|
|
|
renderFooter={(index) => (
|
2024-12-20 20:40:46 +08:00
|
|
|
|
<View style={tailwind("flex flex-row items-center")}>
|
|
|
|
|
<TouchableOpacity
|
|
|
|
|
onPress={() => hanldSaveImage(index)}
|
|
|
|
|
style={{
|
|
|
|
|
marginLeft: 20,
|
|
|
|
|
marginBottom: insets.bottom,
|
|
|
|
|
...tailwind(
|
|
|
|
|
"flex justify-center items-center w-14 h-14 bg-[#FFFFFF1A] rounded-full"
|
|
|
|
|
),
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{isSaving && <ActivityIndicator size="small" />}
|
|
|
|
|
{!isSaving && (
|
|
|
|
|
<Icon
|
|
|
|
|
type="ionicon"
|
|
|
|
|
name="save-outline"
|
|
|
|
|
size={20}
|
|
|
|
|
color="white"
|
|
|
|
|
/>
|
|
|
|
|
)}
|
|
|
|
|
<Text
|
|
|
|
|
style={tailwind("text-xs text-[#FFFFFF80] font-medium mt-1")}
|
|
|
|
|
>
|
|
|
|
|
保存
|
|
|
|
|
</Text>
|
|
|
|
|
</TouchableOpacity>
|
|
|
|
|
{imageUrls[index]?.id && (
|
|
|
|
|
<TouchableOpacity
|
|
|
|
|
onPress={() => handleCheckOriginalImage(index)}
|
|
|
|
|
style={{
|
|
|
|
|
marginLeft: 20,
|
|
|
|
|
marginBottom: insets.bottom,
|
|
|
|
|
...tailwind(
|
|
|
|
|
"flex justify-center items-center w-14 h-14 bg-[#FFFFFF1A] rounded-full"
|
|
|
|
|
),
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<Icon
|
|
|
|
|
type="ionicon"
|
|
|
|
|
name="image-outline"
|
|
|
|
|
size={20}
|
|
|
|
|
color="white"
|
|
|
|
|
/>
|
|
|
|
|
<Text
|
|
|
|
|
style={tailwind("text-xs text-[#FFFFFF80] font-medium mt-1")}
|
|
|
|
|
>
|
|
|
|
|
原图
|
|
|
|
|
</Text>
|
|
|
|
|
</TouchableOpacity>
|
2024-05-27 13:21:36 +08:00
|
|
|
|
)}
|
2024-12-20 20:40:46 +08:00
|
|
|
|
</View>
|
2024-05-27 13:21:36 +08:00
|
|
|
|
)}
|
2024-04-24 16:33:47 +08:00
|
|
|
|
onSave={(url) => {
|
|
|
|
|
saveImage(url);
|
2024-03-15 20:14:22 +08:00
|
|
|
|
}}
|
2024-12-20 20:40:46 +08:00
|
|
|
|
saveToLocalByLongPress={false}
|
2024-03-15 20:14:22 +08:00
|
|
|
|
loadingRender={() => <ActivityIndicator size="large" />}
|
|
|
|
|
/>
|
|
|
|
|
<Toast />
|
|
|
|
|
<MyModal
|
|
|
|
|
visible={isVipModalVisible}
|
|
|
|
|
setVisible={setIsVipModalVisible}
|
|
|
|
|
title="是否开通会员?"
|
2024-12-20 20:40:46 +08:00
|
|
|
|
content="会员可无限制保存图片、查看原图,一次开通永久有效。"
|
2024-03-15 20:14:22 +08:00
|
|
|
|
cancel={() => {
|
|
|
|
|
setIsVipModalVisible(false);
|
|
|
|
|
}}
|
|
|
|
|
confirm={() => {
|
|
|
|
|
setIsVipModalVisible(false);
|
2024-04-23 22:22:18 +08:00
|
|
|
|
closeImageViewer();
|
2024-03-15 20:14:22 +08:00
|
|
|
|
navigation.navigate("WebWithoutHeader", {
|
|
|
|
|
uri: process.env.EXPO_PUBLIC_WEB_URL + "/vip",
|
|
|
|
|
});
|
|
|
|
|
}}
|
|
|
|
|
/>
|
|
|
|
|
</Modal>
|
|
|
|
|
);
|
|
|
|
|
}
|