signal-desktop/ts/state/ducks/preferredReactions.ts
2021-11-11 16:43:05 -06:00

340 lines
9.7 KiB
TypeScript

// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ThunkAction } from 'redux-thunk';
import { omit } from 'lodash';
import * as log from '../../logging/log';
import * as Errors from '../../types/errors';
import { replaceIndex } from '../../util/replaceIndex';
import { useBoundActions } from '../../hooks/useBoundActions';
import type { StateType as RootStateType } from '../reducer';
import { DEFAULT_PREFERRED_REACTION_EMOJI_SHORT_NAMES } from '../../reactions/constants';
import { getPreferredReactionEmoji } from '../../reactions/preferredReactionEmoji';
import { getEmojiSkinTone } from '../selectors/items';
import { convertShortName } from '../../components/emoji/lib';
// State
export type PreferredReactionsStateType = {
customizePreferredReactionsModal?: {
draftPreferredReactions: Array<string>;
originalPreferredReactions: Array<string>;
selectedDraftEmojiIndex: undefined | number;
} & (
| { isSaving: true; hadSaveError: false }
| { isSaving: false; hadSaveError: boolean }
);
};
// Actions
const CANCEL_CUSTOMIZE_PREFERRED_REACTIONS_MODAL =
'preferredReactions/CANCEL_CUSTOMIZE_PREFERRED_REACTIONS_MODAL';
const DESELECT_DRAFT_EMOJI = 'preferredReactions/DESELECT_DRAFT_EMOJI';
const OPEN_CUSTOMIZE_PREFERRED_REACTIONS_MODAL =
'preferredReactions/OPEN_CUSTOMIZE_PREFERRED_REACTIONS_MODAL';
const REPLACE_SELECTED_DRAFT_EMOJI =
'preferredReactions/REPLACE_SELECTED_DRAFT_EMOJI';
const RESET_DRAFT_EMOJI = 'preferredReactions/RESET_DRAFT_EMOJI';
const SAVE_PREFERRED_REACTIONS_FULFILLED =
'preferredReactions/SAVE_PREFERRED_REACTIONS_FULFILLED';
const SAVE_PREFERRED_REACTIONS_PENDING =
'preferredReactions/SAVE_PREFERRED_REACTIONS_PENDING';
const SAVE_PREFERRED_REACTIONS_REJECTED =
'preferredReactions/SAVE_PREFERRED_REACTIONS_REJECTED';
const SELECT_DRAFT_EMOJI_TO_BE_REPLACED =
'preferredReactions/SELECT_DRAFT_EMOJI_TO_BE_REPLACED';
type CancelCustomizePreferredReactionsModalActionType = {
type: typeof CANCEL_CUSTOMIZE_PREFERRED_REACTIONS_MODAL;
};
type DeselectDraftEmojiActionType = { type: typeof DESELECT_DRAFT_EMOJI };
type OpenCustomizePreferredReactionsModalActionType = {
type: typeof OPEN_CUSTOMIZE_PREFERRED_REACTIONS_MODAL;
payload: {
originalPreferredReactions: Array<string>;
};
};
type ReplaceSelectedDraftEmojiActionType = {
type: typeof REPLACE_SELECTED_DRAFT_EMOJI;
payload: string;
};
type ResetDraftEmojiActionType = {
type: typeof RESET_DRAFT_EMOJI;
payload: { skinTone: number };
};
type SavePreferredReactionsFulfilledActionType = {
type: typeof SAVE_PREFERRED_REACTIONS_FULFILLED;
};
type SavePreferredReactionsPendingActionType = {
type: typeof SAVE_PREFERRED_REACTIONS_PENDING;
};
type SavePreferredReactionsRejectedActionType = {
type: typeof SAVE_PREFERRED_REACTIONS_REJECTED;
};
type SelectDraftEmojiToBeReplacedActionType = {
type: typeof SELECT_DRAFT_EMOJI_TO_BE_REPLACED;
payload: number;
};
// Action creators
export const actions = {
cancelCustomizePreferredReactionsModal,
deselectDraftEmoji,
openCustomizePreferredReactionsModal,
replaceSelectedDraftEmoji,
resetDraftEmoji,
savePreferredReactions,
selectDraftEmojiToBeReplaced,
};
export const useActions = (): typeof actions => useBoundActions(actions);
function cancelCustomizePreferredReactionsModal(): CancelCustomizePreferredReactionsModalActionType {
return { type: CANCEL_CUSTOMIZE_PREFERRED_REACTIONS_MODAL };
}
function deselectDraftEmoji(): DeselectDraftEmojiActionType {
return { type: DESELECT_DRAFT_EMOJI };
}
function openCustomizePreferredReactionsModal(): ThunkAction<
void,
RootStateType,
unknown,
OpenCustomizePreferredReactionsModalActionType
> {
return (dispatch, getState) => {
const state = getState();
const originalPreferredReactions = getPreferredReactionEmoji(
getState().items.preferredReactionEmoji,
getEmojiSkinTone(state)
);
dispatch({
type: OPEN_CUSTOMIZE_PREFERRED_REACTIONS_MODAL,
payload: { originalPreferredReactions },
});
};
}
function replaceSelectedDraftEmoji(
newEmoji: string
): ReplaceSelectedDraftEmojiActionType {
return {
type: REPLACE_SELECTED_DRAFT_EMOJI,
payload: newEmoji,
};
}
function resetDraftEmoji(): ThunkAction<
void,
RootStateType,
unknown,
ResetDraftEmojiActionType
> {
return (dispatch, getState) => {
const skinTone = getEmojiSkinTone(getState());
dispatch({ type: RESET_DRAFT_EMOJI, payload: { skinTone } });
};
}
function savePreferredReactions(): ThunkAction<
void,
RootStateType,
unknown,
| SavePreferredReactionsFulfilledActionType
| SavePreferredReactionsPendingActionType
| SavePreferredReactionsRejectedActionType
> {
return async (dispatch, getState) => {
const { draftPreferredReactions } =
getState().preferredReactions.customizePreferredReactionsModal || {};
if (!draftPreferredReactions) {
log.error(
"savePreferredReactions won't work because the modal is not open"
);
return;
}
let succeeded = false;
dispatch({ type: SAVE_PREFERRED_REACTIONS_PENDING });
try {
await window.storage.put(
'preferredReactionEmoji',
draftPreferredReactions
);
succeeded = true;
} catch (err: unknown) {
log.warn(Errors.toLogFormat(err));
}
if (succeeded) {
dispatch({ type: SAVE_PREFERRED_REACTIONS_FULFILLED });
window.ConversationController.getOurConversationOrThrow().captureChange(
'preferredReactionEmoji'
);
} else {
dispatch({ type: SAVE_PREFERRED_REACTIONS_REJECTED });
}
};
}
function selectDraftEmojiToBeReplaced(
index: number
): SelectDraftEmojiToBeReplacedActionType {
return {
type: SELECT_DRAFT_EMOJI_TO_BE_REPLACED,
payload: index,
};
}
// Reducer
export function getInitialState(): PreferredReactionsStateType {
return {};
}
export function reducer(
state: Readonly<PreferredReactionsStateType> = getInitialState(),
action: Readonly<
| CancelCustomizePreferredReactionsModalActionType
| DeselectDraftEmojiActionType
| OpenCustomizePreferredReactionsModalActionType
| ReplaceSelectedDraftEmojiActionType
| ResetDraftEmojiActionType
| SavePreferredReactionsFulfilledActionType
| SavePreferredReactionsPendingActionType
| SavePreferredReactionsRejectedActionType
| SelectDraftEmojiToBeReplacedActionType
>
): PreferredReactionsStateType {
switch (action.type) {
case CANCEL_CUSTOMIZE_PREFERRED_REACTIONS_MODAL:
case SAVE_PREFERRED_REACTIONS_FULFILLED:
return omit(state, ['customizePreferredReactionsModal']);
case DESELECT_DRAFT_EMOJI:
if (!state.customizePreferredReactionsModal) {
return state;
}
return {
...state,
customizePreferredReactionsModal: {
...state.customizePreferredReactionsModal,
selectedDraftEmojiIndex: undefined,
},
};
case OPEN_CUSTOMIZE_PREFERRED_REACTIONS_MODAL: {
const { originalPreferredReactions } = action.payload;
return {
...state,
customizePreferredReactionsModal: {
draftPreferredReactions: originalPreferredReactions,
originalPreferredReactions,
selectedDraftEmojiIndex: undefined,
isSaving: false,
hadSaveError: false,
},
};
}
case REPLACE_SELECTED_DRAFT_EMOJI: {
const newEmoji = action.payload;
const { customizePreferredReactionsModal } = state;
if (!customizePreferredReactionsModal) {
return state;
}
const { draftPreferredReactions, selectedDraftEmojiIndex } =
customizePreferredReactionsModal;
if (selectedDraftEmojiIndex === undefined) {
return state;
}
return {
...state,
customizePreferredReactionsModal: {
...customizePreferredReactionsModal,
draftPreferredReactions: replaceIndex(
draftPreferredReactions,
selectedDraftEmojiIndex,
newEmoji
),
selectedDraftEmojiIndex: undefined,
},
};
}
case RESET_DRAFT_EMOJI: {
const { skinTone } = action.payload;
if (!state.customizePreferredReactionsModal) {
return state;
}
return {
...state,
customizePreferredReactionsModal: {
...state.customizePreferredReactionsModal,
draftPreferredReactions:
DEFAULT_PREFERRED_REACTION_EMOJI_SHORT_NAMES.map(shortName =>
convertShortName(shortName, skinTone)
),
selectedDraftEmojiIndex: undefined,
},
};
}
case SAVE_PREFERRED_REACTIONS_PENDING:
if (!state.customizePreferredReactionsModal) {
return state;
}
return {
...state,
customizePreferredReactionsModal: {
...state.customizePreferredReactionsModal,
selectedDraftEmojiIndex: undefined,
isSaving: true,
hadSaveError: false,
},
};
case SAVE_PREFERRED_REACTIONS_REJECTED:
if (!state.customizePreferredReactionsModal) {
return state;
}
return {
...state,
customizePreferredReactionsModal: {
...state.customizePreferredReactionsModal,
isSaving: false,
hadSaveError: true,
},
};
case SELECT_DRAFT_EMOJI_TO_BE_REPLACED: {
const index = action.payload;
if (
!state.customizePreferredReactionsModal ||
!(
index in
state.customizePreferredReactionsModal.draftPreferredReactions
)
) {
return state;
}
return {
...state,
customizePreferredReactionsModal: {
...state.customizePreferredReactionsModal,
selectedDraftEmojiIndex: index,
},
};
}
default:
return state;
}
}