tiefen_space_app/components/MediaPicker/index.jsx

197 lines
5.4 KiB
React
Raw Normal View History

2023-12-29 00:27:44 +08:00
import { View, TouchableOpacity, Image } from "react-native";
2024-09-27 16:04:56 +08:00
import React, { useState, useEffect, useCallback } from "react";
2023-12-29 00:27:44 +08:00
import { useTailwind } from "tailwind-rn";
import { Icon } from "@rneui/themed";
import Toast from "react-native-toast-message";
import { DraggableGrid } from "react-native-draggable-grid";
import { useImageViewer } from "../../context/ImageViewProvider";
import VideoModal from "../VideoModal";
2024-09-20 15:59:27 +08:00
import * as ImagePicker from "expo-image-picker";
2024-09-27 16:04:56 +08:00
import * as VideoThumbnails from "expo-video-thumbnails";
2023-12-29 00:27:44 +08:00
/*
props格式
setDragging 返回正在拖动的状态用于该组件嵌套在ScrollView中的时候禁用ScrollView的滚动
2023-12-29 00:27:44 +08:00
type 选择的媒体类型 "image"|"video"|"mix"
maxCount 最大选择数量
setAssets 向父组件返回媒体
*/
export default function MediaPicker({
setDragging = () => null,
maxCount,
2024-09-20 15:59:27 +08:00
type = "image",
setAssets,
}) {
2023-12-29 00:27:44 +08:00
const tailwind = useTailwind();
const [newAssets, setNewAssets] = useState([]);
const [_assets, set_Assets] = useState([]);
const { showImageViewer } = useImageViewer();
const [showVideo, setShowVideo] = useState(false);
const [videoUrl, setVideoUrl] = useState("");
2024-09-27 16:04:56 +08:00
//为视频生成封面
const generateThumbnail = useCallback(async (uri) => {
try {
const videoCover = await VideoThumbnails.getThumbnailAsync(uri);
return videoCover;
} catch (e) {
console.warn(e);
}
}, []);
2024-09-20 15:59:27 +08:00
//选择媒体
const pickMedia = async () => {
let mediaType;
if (type === "image") {
mediaType = ImagePicker.MediaTypeOptions.Images;
} else if (type === "video") {
mediaType = ImagePicker.MediaTypeOptions.Videos;
} else {
mediaType = ImagePicker.MediaTypeOptions.All;
}
let result = await ImagePicker.launchImageLibraryAsync({
mediaTypes: mediaType,
allowsMultipleSelection: true,
selectionLimit: maxCount - _assets.length,
quality: 1,
});
if (!result.canceled) {
2024-09-27 16:04:56 +08:00
for (let i = 0; i < result.assets.length; i++) {
if (result.assets[i].duration > 0) {
const videoCover = await generateThumbnail(result.assets[i].uri);
result.assets[i].cover = videoCover.uri;
}
}
2024-09-20 15:59:27 +08:00
setNewAssets(result.assets);
}
};
2023-12-29 00:27:44 +08:00
//当newAssets改变的时候添加新数据到assets
useEffect(() => {
const temAssets = newAssets.map((item) => ({
...item,
key: Date.now().toString(36) + Math.random().toString(36),
}));
set_Assets([..._assets, ...temAssets]);
setAssets([..._assets, ...temAssets]);
2023-12-29 00:27:44 +08:00
}, [newAssets]);
//加号选择器
const Picker = () => {
return (
<TouchableOpacity
onPress={() => {
//数量检查
maxCount - _assets.length !== 0
2024-09-20 15:59:27 +08:00
? pickMedia()
2023-12-29 00:27:44 +08:00
: Toast.show({
type: "error",
text1: `已达到最大选择数量`,
topOffset: 60,
});
}}
style={{
aspectRatio: 1,
padding: 2,
...tailwind("w-1/4"),
2023-12-29 00:27:44 +08:00
}}
>
<View
style={tailwind(
"border border-dashed border-gray-400 rounded flex-1 justify-center items-center"
2023-12-29 00:27:44 +08:00
)}
>
<Icon type="ionicon" name="add" color="#9ca3af" size={40} />
</View>
</TouchableOpacity>
);
};
const renderItem = (item) => {
2023-12-29 00:27:44 +08:00
const handleDelete = () => {
const updatedAssets = _assets.filter((_item) => _item.key !== item.key);
2023-12-29 00:27:44 +08:00
set_Assets(updatedAssets);
setAssets(updatedAssets);
};
2023-12-29 00:27:44 +08:00
return (
<View
activeOpacity={1}
2023-12-29 00:27:44 +08:00
style={{
aspectRatio: 1,
padding: 2,
...tailwind("w-full"),
2023-12-29 00:27:44 +08:00
}}
key={item.key}
2023-12-29 00:27:44 +08:00
>
2024-09-27 16:04:56 +08:00
<Image
src={item.cover ? item.cover : item.uri}
style={tailwind("w-full h-full rounded")}
/>
<View style={{ zIndex: 999, ...tailwind("absolute top-2 right-2") }}>
2023-12-29 00:27:44 +08:00
<Icon
type="ionicon"
name="close"
color="white"
size={18}
onPress={handleDelete}
style={tailwind("bg-black rounded-full opacity-70")}
/>
</View>
2024-09-20 15:59:27 +08:00
{item.duration > 0 && (
<View
style={tailwind(
"absolute flex w-full h-full items-center justify-center"
)}
>
<Icon
type="ionicon"
name="play"
color="white"
size={14}
style={tailwind("bg-black p-2 rounded-full opacity-70")}
/>
</View>
)}
2023-12-29 00:27:44 +08:00
</View>
);
};
return (
<View style={tailwind("flex flex-col w-full")}>
<DraggableGrid
numColumns={4}
renderItem={renderItem}
data={_assets}
onDragStart={() => setDragging(true)}
onDragRelease={(data) => {
set_Assets(data);
setAssets(data);
setDragging(false);
}}
onItemPress={(item) => {
2024-09-20 15:59:27 +08:00
if (!item.duration) {
showImageViewer({
2024-09-20 15:59:27 +08:00
imageUrls: [{ url: item.uri }],
index: 0,
});
return;
}
setVideoUrl(item.uri);
setShowVideo(true);
}}
/>
2023-12-29 00:27:44 +08:00
<Picker />
<VideoModal
visible={showVideo}
setVisible={setShowVideo}
url={videoUrl}
/>
2023-12-29 00:27:44 +08:00
</View>
);
}