databag/app/mobile/src/session/conversation/topicItem/imageAsset/useImageAsset.hook.js
2024-06-10 13:33:58 -07:00

128 lines
3.9 KiB
JavaScript

import { useState, useRef, useEffect, useContext } from 'react';
import { ConversationContext } from 'context/ConversationContext';
import { Image, Platform } from 'react-native';
import { useWindowDimensions } from 'react-native';
import RNFS from "react-native-fs";
export function useImageAsset(asset) {
const [state, setState] = useState({
frameWidth: 1,
frameHeight: 1,
imageRatio: 1,
imageWidth: 1024,
imageHeight: 1024,
url: null,
loaded: false,
failed: false,
controls: false,
downloaded: false,
showDownloaded: false,
});
const conversation = useContext(ConversationContext);
const dimensions = useWindowDimensions();
const controls = useRef();
const updateState = (value) => {
setState((s) => ({ ...s, ...value }));
}
useEffect(() => {
if (state.loaded) {
const frameRatio = state.frameWidth / state.frameHeight;
if (frameRatio > state.imageRatio) {
//height constrained
const height = Math.floor(0.9 * state.frameHeight);
const width = Math.floor(height * state.imageRatio);
updateState({ imageWidth: width, imageHeight: height });
}
else {
//width constrained
const width = Math.floor(0.9 * state.frameWidth);
const height = Math.floor(width / state.imageRatio);
updateState({ imageWidth: width, imageHeight: height });
}
}
actions.showControls();
}, [state.frameWidth, state.frameHeight, state.imageRatio, state.loaded]);
useEffect(() => {
imageWidth = dimensions.width * 0.9 > state.imageWidth ? state.imageWidth : dimensions.width * 0.9;
imageHeight = dimensions.height * 0.9 > state.imageHeight ? state.imageHeight : dimensions.height * 0.9;
updateState({ frameWidth: dimensions.width, frameHeight: dimensions.height, imageWidth, imageHeight });
}, [dimensions]);
useEffect(() => {
if (asset.encrypted) {
const now = Date.now();
const url = asset.decrypted ? `file://${asset.decrypted}?now=${now}` : null
updateState({ url, failed: asset.error });
}
else {
updateState({ url: asset.full, failed: false });
}
}, [asset]);
const actions = {
setRatio: (e) => {
const { width, height } = e.nativeEvent;
updateState({ imageRatio: width / height });
},
download: async () => {
if (!state.downloaded) {
updateState({ downloaded: true });
const epoch = Math.ceil(Date.now() / 1000);
const dir = Platform.OS === 'ios' ? RNFS.DocumentDirectoryPath : RNFS.DownloadDirectoryPath;
const path = `${dir}/databag_${epoch}`
if (state.url.substring(0, 7) === 'file://') {
await RNFS.copyFile(state.url.substring(7).split('?')[0], path);
}
else {
await RNFS.downloadFile({ fromUrl: state.url, toFile: path }).promise;
}
let ext = 'dat';
const block = await RNFS.read(path, 8, 0, 'base64');
if (block === '/9j/4AAQSkY=') {
ext = 'jpg';
}
if (block === 'iVBORw0KGgo=') {
ext = 'png';
}
if (block === 'UklGRphXAQA=') {
ext = 'webp';
}
if (block === 'R0lGODlhIAM=') {
ext = 'gif';
}
else if (block.startsWith('Qk')) {
ext = 'bmp';
}
await RNFS.moveFile(path, `${path}.${ext}`);
if (Platform.OS !== 'ios') {
await RNFS.scanFile(`${path}.${ext}`);
}
updateState({ showDownloaded: true });
setTimeout(() => { updateState({ showDownloaded: false }) }, 2000);
}
},
loaded: () => {
updateState({ loaded: true });
},
failed: (e) => {
updateState({ failed: true });
},
showControls: () => {
clearTimeout(controls.current);
updateState({ controls: true });
controls.current = setTimeout(() => {
updateState({ controls: false });
}, 2000);
},
};
return { state, actions };
}