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";
|
2024-05-29 16:39:34 +08:00
|
|
|
|
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格式:
|
2024-05-29 16:39:34 +08:00
|
|
|
|
setDragging 返回正在拖动的状态,用于该组件嵌套在ScrollView中的时候禁用ScrollView的滚动
|
2023-12-29 00:27:44 +08:00
|
|
|
|
type 选择的媒体类型 "image"|"video"|"mix"
|
|
|
|
|
maxCount 最大选择数量
|
|
|
|
|
setAssets 向父组件返回媒体
|
|
|
|
|
*/
|
|
|
|
|
|
2024-05-29 16:39:34 +08:00
|
|
|
|
export default function MediaPicker({
|
|
|
|
|
setDragging = () => null,
|
|
|
|
|
maxCount,
|
2024-09-20 15:59:27 +08:00
|
|
|
|
type = "image",
|
2024-05-29 16:39:34 +08:00
|
|
|
|
setAssets,
|
|
|
|
|
}) {
|
2023-12-29 00:27:44 +08:00
|
|
|
|
const tailwind = useTailwind();
|
|
|
|
|
const [newAssets, setNewAssets] = useState([]);
|
|
|
|
|
const [_assets, set_Assets] = useState([]);
|
|
|
|
|
|
2024-05-29 16:39:34 +08:00
|
|
|
|
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(() => {
|
2024-05-29 16:39:34 +08:00
|
|
|
|
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,
|
2024-05-29 16:39:34 +08:00
|
|
|
|
padding: 2,
|
|
|
|
|
...tailwind("w-1/4"),
|
2023-12-29 00:27:44 +08:00
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
<View
|
|
|
|
|
style={tailwind(
|
2024-05-29 16:39:34 +08:00
|
|
|
|
"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>
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
2024-05-29 16:39:34 +08:00
|
|
|
|
const renderItem = (item) => {
|
2023-12-29 00:27:44 +08:00
|
|
|
|
const handleDelete = () => {
|
2024-05-29 16:39:34 +08:00
|
|
|
|
const updatedAssets = _assets.filter((_item) => _item.key !== item.key);
|
2023-12-29 00:27:44 +08:00
|
|
|
|
set_Assets(updatedAssets);
|
|
|
|
|
setAssets(updatedAssets);
|
|
|
|
|
};
|
2024-05-29 16:39:34 +08:00
|
|
|
|
|
2023-12-29 00:27:44 +08:00
|
|
|
|
return (
|
|
|
|
|
<View
|
2024-05-29 16:39:34 +08:00
|
|
|
|
activeOpacity={1}
|
2023-12-29 00:27:44 +08:00
|
|
|
|
style={{
|
|
|
|
|
aspectRatio: 1,
|
2024-05-29 16:39:34 +08:00
|
|
|
|
padding: 2,
|
|
|
|
|
...tailwind("w-full"),
|
2023-12-29 00:27:44 +08:00
|
|
|
|
}}
|
2024-05-29 16:39:34 +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 && (
|
2024-05-29 16:39:34 +08:00
|
|
|
|
<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 (
|
2024-05-29 16:39:34 +08:00
|
|
|
|
<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) {
|
2024-05-29 16:39:34 +08:00
|
|
|
|
showImageViewer({
|
2024-09-20 15:59:27 +08:00
|
|
|
|
imageUrls: [{ url: item.uri }],
|
2024-05-29 16:39:34 +08:00
|
|
|
|
index: 0,
|
|
|
|
|
});
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
setVideoUrl(item.uri);
|
|
|
|
|
setShowVideo(true);
|
|
|
|
|
}}
|
|
|
|
|
/>
|
2023-12-29 00:27:44 +08:00
|
|
|
|
<Picker />
|
2024-05-29 16:39:34 +08:00
|
|
|
|
<VideoModal
|
|
|
|
|
visible={showVideo}
|
|
|
|
|
setVisible={setShowVideo}
|
|
|
|
|
url={videoUrl}
|
|
|
|
|
/>
|
2023-12-29 00:27:44 +08:00
|
|
|
|
</View>
|
|
|
|
|
);
|
|
|
|
|
}
|