Files
mattermost-mobile/app/screens/custom_status/custom_status.tsx
Daniel Espino García 4e3531fb52 Refactor CustomStatus to functional component (#6899)
* Refactor CustomStatus to functional component

* Fix setting duration to Don't clear

Co-authored-by: Elias Nahum <nahumhbl@gmail.com>
2022-12-23 14:35:33 +02:00

397 lines
16 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import moment from 'moment-timezone';
import React, {useCallback, useEffect, useMemo, useReducer} from 'react';
import {useIntl} from 'react-intl';
import {DeviceEventEmitter, Keyboard, KeyboardAvoidingView, Platform, ScrollView, View} from 'react-native';
import {Edge, SafeAreaView} from 'react-native-safe-area-context';
import {updateLocalCustomStatus} from '@actions/local/user';
import {removeRecentCustomStatus, updateCustomStatus, unsetCustomStatus} from '@actions/remote/user';
import CompassIcon from '@components/compass_icon';
import TabletTitle from '@components/tablet_title';
import {Events, Screens} from '@constants';
import {CustomStatusDurationEnum, SET_CUSTOM_STATUS_FAILURE} from '@constants/custom_status';
import {useServerUrl} from '@context/server';
import {useTheme} from '@context/theme';
import useAndroidHardwareBackHandler from '@hooks/android_back_handler';
import {useIsTablet} from '@hooks/device';
import useNavButtonPressed from '@hooks/navigation_button_pressed';
import {dismissModal, goToScreen, showModal} from '@screens/navigation';
import {getCurrentMomentForTimezone, getRoundedTime} from '@utils/helpers';
import {logDebug} from '@utils/log';
import {mergeNavigationOptions} from '@utils/navigation';
import {preventDoubleTap} from '@utils/tap';
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
import {
getTimezone,
getUserCustomStatus,
isCustomStatusExpired as verifyExpiredStatus,
} from '@utils/user';
import ClearAfter from './components/clear_after';
import CustomStatusInput from './components/custom_status_input';
import CustomStatusSuggestions from './components/custom_status_suggestions';
import RecentCustomStatuses from './components/recent_custom_statuses';
import type UserModel from '@typings/database/models/servers/user';
type NewStatusType = {
emoji?: string;
text?: string;
duration: CustomStatusDuration;
expiresAt: moment.Moment;
}
type Props = {
customStatusExpirySupported: boolean;
currentUser: UserModel;
recentCustomStatuses: UserCustomStatus[];
componentId: string;
}
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => {
return {
container: {
flex: 1,
backgroundColor: changeOpacity(theme.centerChannelColor, 0.03),
},
contentContainerStyle: {
height: '99%',
},
scrollView: {
flex: 1,
paddingTop: 32,
},
separator: {
marginTop: 32,
},
block: {
borderBottomColor: changeOpacity(theme.centerChannelColor, 0.1),
borderBottomWidth: 1,
borderTopColor: changeOpacity(theme.centerChannelColor, 0.1),
borderTopWidth: 1,
},
};
});
const DEFAULT_DURATION: CustomStatusDuration = 'today';
const BTN_UPDATE_STATUS = 'update-custom-status';
const edges: Edge[] = ['bottom', 'left', 'right'];
const calculateExpiryTime = (duration: CustomStatusDuration, currentUser: UserModel, expiresAt: moment.Moment): string => {
const userTimezone = getTimezone(currentUser.timezone);
const currentTime = getCurrentMomentForTimezone(userTimezone);
switch (duration) {
case 'thirty_minutes':
return currentTime.add(30, 'minutes').seconds(0).milliseconds(0).toISOString();
case 'one_hour':
return currentTime.add(1, 'hour').seconds(0).milliseconds(0).toISOString();
case 'four_hours':
return currentTime.add(4, 'hours').seconds(0).milliseconds(0).toISOString();
case 'today':
return currentTime.endOf('day').toISOString();
case 'this_week':
return currentTime.endOf('week').toISOString();
case 'date_and_time':
return expiresAt.toISOString();
case CustomStatusDurationEnum.DONT_CLEAR:
default:
return '';
}
};
function reducer(state: NewStatusType, action: {
type: 'clear' | 'fromUserCustomStatus' | 'fromUserCustomStatusIgnoringExpire' | 'text' | 'emoji' | 'duration';
status?: UserCustomStatus;
value?: string;
duration?: CustomStatusDuration;
expiresAt?: string;
}): NewStatusType {
switch (action.type) {
case 'clear':
return {emoji: '', text: '', duration: DEFAULT_DURATION, expiresAt: state.expiresAt};
case 'fromUserCustomStatus': {
const status = action.status;
if (status) {
return {emoji: status.emoji, text: status.text, duration: status.duration!, expiresAt: moment(status.expires_at)};
}
return state;
}
case 'fromUserCustomStatusIgnoringExpire': {
const status = action.status;
if (status) {
return {emoji: status.emoji, text: status.text, duration: status.duration!, expiresAt: state.expiresAt};
}
return state;
}
case 'text':
return {...state, text: action.value};
case 'emoji':
return {...state, emoji: action.value};
case 'duration':
if (action.duration != null) {
return {
...state,
duration: action.duration,
expiresAt: action.duration === 'date_and_time' && action.expiresAt ? moment(action.expiresAt) : state.expiresAt,
};
}
return state;
default:
return state;
}
}
const CustomStatus = ({
customStatusExpirySupported,
currentUser,
recentCustomStatuses,
componentId,
}: Props) => {
const intl = useIntl();
const isTablet = useIsTablet();
const theme = useTheme();
const style = getStyleSheet(theme);
const serverUrl = useServerUrl();
const storedStatus = useMemo(() => {
return getUserCustomStatus(currentUser);
}, [currentUser]);
const initialStatus = useMemo(() => {
const userTimezone = getTimezone(currentUser.timezone);
// May be a ref
const isCustomStatusExpired = verifyExpiredStatus(currentUser);
const currentTime = getCurrentMomentForTimezone(userTimezone ?? '');
let initialCustomExpiryTime = getRoundedTime(currentTime);
const isCurrentCustomStatusSet = !isCustomStatusExpired && (storedStatus?.text || storedStatus?.emoji);
if (isCurrentCustomStatusSet && storedStatus?.duration === 'date_and_time' && storedStatus?.expires_at) {
initialCustomExpiryTime = moment(storedStatus?.expires_at);
}
return {
duration: isCurrentCustomStatusSet ? storedStatus?.duration ?? CustomStatusDurationEnum.DONT_CLEAR : DEFAULT_DURATION,
emoji: isCurrentCustomStatusSet ? storedStatus?.emoji : '',
expiresAt: initialCustomExpiryTime,
text: isCurrentCustomStatusSet ? storedStatus?.text : '',
};
}, []);
const [newStatus, dispatchStatus] = useReducer(reducer, initialStatus);
const isStatusSet = Boolean(newStatus.emoji || newStatus.text);
const handleClear = useCallback(() => {
dispatchStatus({type: 'clear'});
}, []);
const handleTextChange = useCallback((value: string) => {
dispatchStatus({type: 'text', value});
}, []);
const handleEmojiClick = useCallback((value: string) => {
dispatchStatus({type: 'emoji', value});
}, []);
const handleClearAfterClick = useCallback((duration: CustomStatusDuration, expiresAt: string) => {
dispatchStatus({type: 'duration', duration, expiresAt});
}, []);
const handleRecentCustomStatusClear = useCallback((status: UserCustomStatus) => removeRecentCustomStatus(serverUrl, status), [serverUrl]);
const handleCustomStatusSuggestionClick = useCallback((status: UserCustomStatus) => {
if (!status.duration) {
// This should never happen, but we add a safeguard here
logDebug('clicked on a custom status with no duration');
return;
}
dispatchStatus({type: 'fromUserCustomStatusIgnoringExpire', status});
}, []);
const openClearAfterModal = useCallback(() => {
const screen = Screens.CUSTOM_STATUS_CLEAR_AFTER;
const title = intl.formatMessage({id: 'mobile.custom_status.clear_after.title', defaultMessage: 'Clear Custom Status After'});
const passProps = {
handleClearAfterClick,
initialDuration: newStatus.duration,
intl,
theme,
};
if (isTablet) {
showModal(screen, title, passProps);
} else {
goToScreen(screen, title, passProps);
}
}, [intl, theme, isTablet, newStatus.duration, handleClearAfterClick]);
const handleRecentCustomStatusSuggestionClick = useCallback((status: UserCustomStatus) => {
dispatchStatus({type: 'fromUserCustomStatusIgnoringExpire', status: {...status, duration: status.duration || CustomStatusDurationEnum.DONT_CLEAR}});
if (status.duration === 'date_and_time') {
openClearAfterModal();
}
}, [openClearAfterModal]);
const handleSetStatus = useCallback(async () => {
if (isStatusSet) {
let isStatusSame =
storedStatus?.emoji === newStatus.emoji &&
storedStatus?.text === newStatus.text &&
storedStatus?.duration === newStatus.duration;
const newExpiresAt = calculateExpiryTime(newStatus.duration!, currentUser, newStatus.expiresAt);
if (isStatusSame && newStatus.duration === 'date_and_time') {
isStatusSame = storedStatus?.expires_at === newExpiresAt;
}
if (!isStatusSame) {
const status: UserCustomStatus = {
emoji: newStatus.emoji || 'speech_balloon',
text: newStatus.text?.trim(),
duration: CustomStatusDurationEnum.DONT_CLEAR,
};
if (customStatusExpirySupported) {
status.duration = newStatus.duration;
status.expires_at = newExpiresAt;
}
const {error} = await updateCustomStatus(serverUrl, status);
if (error) {
DeviceEventEmitter.emit(SET_CUSTOM_STATUS_FAILURE);
return;
}
updateLocalCustomStatus(serverUrl, currentUser, status);
dispatchStatus({type: 'fromUserCustomStatus', status});
}
} else if (storedStatus?.emoji) {
const unsetResponse = await unsetCustomStatus(serverUrl);
if (unsetResponse?.data) {
updateLocalCustomStatus(serverUrl, currentUser, undefined);
}
}
Keyboard.dismiss();
if (isTablet) {
DeviceEventEmitter.emit(Events.ACCOUNT_SELECT_TABLET_VIEW, '');
} else {
dismissModal();
}
}, [newStatus, isStatusSet, storedStatus, currentUser]);
const openEmojiPicker = useCallback(preventDoubleTap(() => {
CompassIcon.getImageSource('close', 24, theme.sidebarHeaderTextColor).then((source) => {
const screen = Screens.EMOJI_PICKER;
const title = intl.formatMessage({id: 'mobile.custom_status.choose_emoji', defaultMessage: 'Choose an emoji'});
const passProps = {closeButton: source, onEmojiPress: handleEmojiClick};
showModal(screen, title, passProps);
});
}), [theme, intl, handleEmojiClick]);
const handleBackButton = useCallback(() => {
if (isTablet) {
DeviceEventEmitter.emit(Events.ACCOUNT_SELECT_TABLET_VIEW, '');
} else {
dismissModal({componentId});
}
}, [isTablet]);
useAndroidHardwareBackHandler(componentId, handleBackButton);
useNavButtonPressed(BTN_UPDATE_STATUS, componentId, handleSetStatus, [handleSetStatus]);
useEffect(() => {
mergeNavigationOptions(componentId, {
topBar: {
rightButtons: [
{
enabled: true,
id: BTN_UPDATE_STATUS,
showAsAction: 'always',
testID: 'custom_status.done.button',
text: intl.formatMessage({id: 'mobile.custom_status.modal_confirm', defaultMessage: 'Done'}),
color: theme.sidebarHeaderTextColor,
},
],
},
});
}, []);
return (
<>
{isTablet &&
<TabletTitle
action={intl.formatMessage({id: 'mobile.custom_status.modal_confirm', defaultMessage: 'Done'})}
onPress={handleSetStatus}
testID='custom_status'
title={intl.formatMessage({id: 'mobile.routes.custom_status', defaultMessage: 'Set a custom status'})}
/>
}
<SafeAreaView
edges={edges}
style={style.container}
testID='custom_status.screen'
>
<KeyboardAvoidingView
behavior='padding'
enabled={Platform.OS === 'ios'}
keyboardVerticalOffset={100}
contentContainerStyle={style.contentContainerStyle}
>
<ScrollView
bounces={false}
keyboardDismissMode='none'
keyboardShouldPersistTaps='always'
testID='custom_status.scroll_view'
>
<View style={style.scrollView}>
<View style={style.block}>
<CustomStatusInput
emoji={newStatus.emoji}
isStatusSet={isStatusSet}
onChangeText={handleTextChange}
onClearHandle={handleClear}
onOpenEmojiPicker={openEmojiPicker}
text={newStatus.text}
theme={theme}
/>
{isStatusSet && customStatusExpirySupported && (
<ClearAfter
duration={newStatus.duration}
expiresAt={newStatus.expiresAt}
onOpenClearAfterModal={openClearAfterModal}
theme={theme}
/>
)}
</View>
{recentCustomStatuses.length > 0 && (
<RecentCustomStatuses
onHandleClear={handleRecentCustomStatusClear}
onHandleSuggestionClick={handleRecentCustomStatusSuggestionClick}
recentCustomStatuses={recentCustomStatuses}
theme={theme}
/>
)
}
<CustomStatusSuggestions
intl={intl}
onHandleCustomStatusSuggestionClick={handleCustomStatusSuggestionClick}
recentCustomStatuses={recentCustomStatuses}
theme={theme}
/>
</View>
<View style={style.separator}/>
</ScrollView>
</KeyboardAvoidingView>
</SafeAreaView>
</>
);
};
export default CustomStatus;