159 lines
4.2 KiB
TypeScript
159 lines
4.2 KiB
TypeScript
import { Alert } from 'react-native';
|
|
import DocumentPicker from 'react-native-document-picker';
|
|
import ImagePicker, { ImageOrVideo } from 'react-native-image-crop-picker';
|
|
|
|
import { IMAGE_PICKER_CONFIG, LIBRARY_PICKER_CONFIG, VIDEO_PICKER_CONFIG } from '../constants';
|
|
import { forceJpgExtension } from '../helpers';
|
|
import I18n from '../../../i18n';
|
|
import { canUploadFile } from '../../../lib/methods/helpers';
|
|
import log from '../../../lib/methods/helpers/log';
|
|
import { getSubscriptionByRoomId } from '../../../lib/database/services/Subscription';
|
|
import { getThreadById } from '../../../lib/database/services/Thread';
|
|
import Navigation from '../../../lib/navigation/appNavigation';
|
|
import { useAppSelector } from '../../../lib/hooks';
|
|
import { useRoomContext } from '../../../views/RoomView/context';
|
|
|
|
export const useChooseMedia = ({
|
|
rid,
|
|
tmid,
|
|
permissionToUpload
|
|
}: {
|
|
rid?: string;
|
|
tmid?: string;
|
|
permissionToUpload: boolean;
|
|
}) => {
|
|
const { FileUpload_MediaTypeWhiteList, FileUpload_MaxFileSize } = useAppSelector(state => state.settings);
|
|
const { action, setQuotesAndText, selectedMessages, getText } = useRoomContext();
|
|
const allowList = FileUpload_MediaTypeWhiteList as string;
|
|
const maxFileSize = FileUpload_MaxFileSize as number;
|
|
const libPickerLabels = {
|
|
cropperChooseText: I18n.t('Choose'),
|
|
cropperCancelText: I18n.t('Cancel'),
|
|
loadingLabelText: I18n.t('Processing')
|
|
};
|
|
|
|
const takePhoto = async () => {
|
|
try {
|
|
let image = await ImagePicker.openCamera({ ...IMAGE_PICKER_CONFIG, ...libPickerLabels });
|
|
image = forceJpgExtension(image);
|
|
const file = image as any; // FIXME: unify those types to remove the need for any
|
|
const canUploadResult = canUploadFile({
|
|
file,
|
|
allowList,
|
|
maxFileSize,
|
|
permissionToUploadFile: permissionToUpload
|
|
});
|
|
if (canUploadResult.success) {
|
|
return openShareView([image]);
|
|
}
|
|
|
|
handleError(canUploadResult.error);
|
|
} catch (e) {
|
|
log(e);
|
|
}
|
|
};
|
|
|
|
const takeVideo = async () => {
|
|
try {
|
|
const video = await ImagePicker.openCamera({ ...VIDEO_PICKER_CONFIG, ...libPickerLabels });
|
|
const file = video as any; // FIXME: unify those types to remove the need for any
|
|
const canUploadResult = canUploadFile({
|
|
file,
|
|
allowList,
|
|
maxFileSize,
|
|
permissionToUploadFile: permissionToUpload
|
|
});
|
|
if (canUploadResult.success) {
|
|
return openShareView([video]);
|
|
}
|
|
|
|
handleError(canUploadResult.error);
|
|
} catch (e) {
|
|
log(e);
|
|
}
|
|
};
|
|
|
|
const chooseFromLibrary = async () => {
|
|
try {
|
|
// The type can be video or photo, however the lib understands that it is just one of them.
|
|
let attachments = (await ImagePicker.openPicker({
|
|
...LIBRARY_PICKER_CONFIG,
|
|
...libPickerLabels
|
|
})) as unknown as ImageOrVideo[]; // FIXME: type this
|
|
attachments = attachments.map(att => forceJpgExtension(att));
|
|
openShareView(attachments);
|
|
} catch (e) {
|
|
log(e);
|
|
}
|
|
};
|
|
|
|
const chooseFile = async () => {
|
|
try {
|
|
const res = await DocumentPicker.pickSingle({
|
|
type: [DocumentPicker.types.allFiles]
|
|
});
|
|
const file = {
|
|
filename: res.name,
|
|
size: res.size,
|
|
mime: res.type,
|
|
path: res.uri
|
|
} as any;
|
|
const canUploadResult = canUploadFile({
|
|
file,
|
|
allowList,
|
|
maxFileSize,
|
|
permissionToUploadFile: permissionToUpload
|
|
});
|
|
if (canUploadResult.success) {
|
|
return openShareView([file]);
|
|
}
|
|
handleError(canUploadResult.error);
|
|
} catch (e: any) {
|
|
if (!DocumentPicker.isCancel(e)) {
|
|
log(e);
|
|
}
|
|
}
|
|
};
|
|
|
|
const startShareView = () => {
|
|
const text = getText?.() || '';
|
|
return {
|
|
selectedMessages,
|
|
text
|
|
};
|
|
};
|
|
|
|
const finishShareView = (text = '', quotes = []) => setQuotesAndText?.(text, quotes);
|
|
|
|
const openShareView = async (attachments: any) => {
|
|
if (!rid) return;
|
|
const room = await getSubscriptionByRoomId(rid);
|
|
let thread;
|
|
if (tmid) {
|
|
thread = await getThreadById(tmid);
|
|
}
|
|
if (room) {
|
|
// FIXME: use useNavigation
|
|
Navigation.navigate('ShareView', {
|
|
room,
|
|
thread,
|
|
attachments,
|
|
action,
|
|
finishShareView,
|
|
startShareView
|
|
});
|
|
}
|
|
};
|
|
|
|
const handleError = (error?: string) => {
|
|
Alert.alert(I18n.t('Error_uploading'), error && I18n.isTranslated(error) ? I18n.t(error) : error);
|
|
};
|
|
|
|
return {
|
|
takePhoto,
|
|
takeVideo,
|
|
chooseFromLibrary,
|
|
chooseFile
|
|
};
|
|
};
|