Rocket.Chat.ReactNative/app/containers/MessageComposer/hooks/useChooseMedia.ts

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
};
};