Files
mattermost-mobile/app/utils/gallery/index.ts
Daniel Espino García f4e6917185 Ensure no unresolved types in the definition files (#6521)
* Ensure no unresolved types in the definition files

* Address feedback and general cleanup

* Move import from @constants/x to @constants where relevant

* Remove unneeded "import as"
2022-08-05 14:36:19 +02:00

177 lines
4.7 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import React from 'react';
import {DeviceEventEmitter, Keyboard, NativeModules, Platform} from 'react-native';
import {Navigation, Options, OptionsLayout} from 'react-native-navigation';
import {measure} from 'react-native-reanimated';
import {Events, Screens} from '@constants';
import {showOverlay} from '@screens/navigation';
import {isImage, isVideo} from '@utils/file';
import {generateId} from '@utils/general';
import type {GalleryItemType, GalleryManagerSharedValues} from '@typings/screens/gallery';
export const clamp = (value: number, lowerBound: number, upperBound: number) => {
'worklet';
return Math.min(Math.max(lowerBound, value), upperBound);
};
export const clampVelocity = (velocity: number, minVelocity: number, maxVelocity: number) => {
'worklet';
if (velocity > 0) {
return Math.min(Math.max(velocity, minVelocity), maxVelocity);
}
return Math.max(Math.min(velocity, -minVelocity), -maxVelocity);
};
export const fileToGalleryItem = (file: FileInfo, authorId?: string): GalleryItemType => {
let type: GalleryItemType['type'] = 'file';
if (isVideo(file)) {
type = 'video';
} else if (isImage(file)) {
type = 'image';
}
return {
authorId,
extension: file.extension,
height: file.height,
id: file.id || generateId('uid'),
mime_type: file.mime_type,
name: file.name,
posterUri: type === 'video' ? file.mini_preview : undefined, // set the video poster to the mini_preview
postId: file.post_id,
size: file.size,
type,
uri: file.localPath || file.uri || '',
width: file.width,
};
};
export const freezeOtherScreens = (value: boolean) => {
DeviceEventEmitter.emit(Events.FREEZE_SCREEN, value);
};
export const friction = (value: number) => {
'worklet';
const MAX_FRICTION = 30;
const MAX_VALUE = 200;
const res = Math.max(
1,
Math.min(
MAX_FRICTION,
(1 + (Math.abs(value) * (MAX_FRICTION - 1))) / MAX_VALUE,
),
);
return value > 0 ? res : -res;
};
export const galleryItemToFileInfo = (item: GalleryItemType): FileInfo => ({
id: item.id,
name: item.name,
create_at: 0,
delete_at: 0,
update_at: 0,
width: item.width,
height: item.height,
extension: item.extension || '',
mime_type: item.mime_type,
has_preview_image: false,
post_id: item.postId!,
size: 0,
user_id: item.authorId!,
});
export const getShouldRender = (index: number, activeIndex: number, diffValue = 3) => {
const diff = Math.abs(index - activeIndex);
if (diff > diffValue) {
return false;
}
return true;
};
export function measureItem(ref: React.RefObject<any>, sharedValues: GalleryManagerSharedValues) {
'worklet';
try {
const measurements = measure(ref);
sharedValues.x.value = measurements.pageX;
sharedValues.y.value = measurements.pageY;
sharedValues.width.value = measurements.width;
sharedValues.height.value = measurements.height;
} catch (err) {
sharedValues.x.value = 999999;
sharedValues.y.value = 999999;
sharedValues.width.value = 0;
sharedValues.height.value = 0;
}
}
export function openGalleryAtIndex(galleryIdentifier: string, initialIndex: number, items: GalleryItemType[], hideActions = false) {
Keyboard.dismiss();
const props = {
galleryIdentifier,
hideActions,
initialIndex,
items,
};
const layout: OptionsLayout = {
orientation: ['portrait', 'landscape'],
};
const options: Options = {
layout,
topBar: {
background: {
color: '#000',
},
visible: Platform.OS === 'android',
},
statusBar: {
backgroundColor: '#000',
style: 'light',
},
animations: {
showModal: {
waitForRender: false,
enabled: false,
},
dismissModal: {
enabled: false,
},
},
};
if (Platform.OS === 'ios') {
// on iOS we need both the navigation & the module
Navigation.setDefaultOptions({layout});
NativeModules.MattermostManaged.unlockOrientation();
}
showOverlay(Screens.GALLERY, props, options);
setTimeout(() => {
freezeOtherScreens(true);
}, 500);
}
export const typedMemo: <T>(c: T) => T = React.memo;
export const workletNoop = () => {
'worklet';
};
export const workletNoopTrue = () => {
'worklet';
return true;
};