import { List as ImmutableList } from 'immutable'; import { isLoggedIn } from 'soapbox/utils/auth'; import { getClient } from '../api'; import { importFetchedStatus } from './importer'; import { favourite, unfavourite } from './interactions'; import type { AppDispatch, RootState } from 'soapbox/store'; import type { EmojiReaction, Status } from 'soapbox/types/entities'; const EMOJI_REACT_REQUEST = 'EMOJI_REACT_REQUEST'; const EMOJI_REACT_SUCCESS = 'EMOJI_REACT_SUCCESS'; const EMOJI_REACT_FAIL = 'EMOJI_REACT_FAIL'; const UNEMOJI_REACT_REQUEST = 'UNEMOJI_REACT_REQUEST'; const UNEMOJI_REACT_SUCCESS = 'UNEMOJI_REACT_SUCCESS'; const UNEMOJI_REACT_FAIL = 'UNEMOJI_REACT_FAIL'; const noOp = () => () => new Promise(f => f(undefined)); const simpleEmojiReact = (status: Status, emoji: string, custom?: string) => (dispatch: AppDispatch) => { const emojiReacts: ImmutableList = status.reactions || ImmutableList(); if (emoji === '👍' && status.favourited) return dispatch(unfavourite(status)); const undo = emojiReacts.filter(e => e.me === true && e.name === emoji).count() > 0; if (undo) return dispatch(unEmojiReact(status, emoji)); return Promise.all([ ...emojiReacts .filter((emojiReact) => emojiReact.me === true) .map(emojiReact => dispatch(unEmojiReact(status, emojiReact.name))).toArray(), status.favourited && dispatch(unfavourite(status)), ]).then(() => { if (emoji === '👍') { dispatch(favourite(status)); } else { dispatch(emojiReact(status, emoji, custom)); } }).catch(err => { console.error(err); }); }; const emojiReact = (status: Status, emoji: string, custom?: string) => (dispatch: AppDispatch, getState: () => RootState) => { if (!isLoggedIn(getState)) return dispatch(noOp()); dispatch(emojiReactRequest(status, emoji, custom)); return getClient(getState).statuses.createStatusReaction(status.id, emoji).then((response) => { dispatch(importFetchedStatus(response)); dispatch(emojiReactSuccess(status, emoji)); }).catch((error) => { dispatch(emojiReactFail(status, emoji, error)); }); }; const unEmojiReact = (status: Status, emoji: string) => (dispatch: AppDispatch, getState: () => RootState) => { if (!isLoggedIn(getState)) return dispatch(noOp()); dispatch(unEmojiReactRequest(status, emoji)); return getClient(getState).statuses.deleteStatusReaction(status.id, emoji).then(response => { dispatch(importFetchedStatus(response)); dispatch(unEmojiReactSuccess(status, emoji)); }).catch(error => { dispatch(unEmojiReactFail(status, emoji, error)); }); }; const emojiReactRequest = (status: Status, emoji: string, custom?: string) => ({ type: EMOJI_REACT_REQUEST, status, emoji, custom, skipLoading: true, }); const emojiReactSuccess = (status: Status, emoji: string) => ({ type: EMOJI_REACT_SUCCESS, status, emoji, skipLoading: true, }); const emojiReactFail = (status: Status, emoji: string, error: unknown) => ({ type: EMOJI_REACT_FAIL, status, emoji, error, skipLoading: true, }); const unEmojiReactRequest = (status: Status, emoji: string) => ({ type: UNEMOJI_REACT_REQUEST, status, emoji, skipLoading: true, }); const unEmojiReactSuccess = (status: Status, emoji: string) => ({ type: UNEMOJI_REACT_SUCCESS, status, emoji, skipLoading: true, }); const unEmojiReactFail = (status: Status, emoji: string, error: unknown) => ({ type: UNEMOJI_REACT_FAIL, status, emoji, error, skipLoading: true, }); export { EMOJI_REACT_REQUEST, EMOJI_REACT_SUCCESS, EMOJI_REACT_FAIL, UNEMOJI_REACT_REQUEST, UNEMOJI_REACT_SUCCESS, UNEMOJI_REACT_FAIL, simpleEmojiReact, emojiReact, unEmojiReact, emojiReactRequest, emojiReactSuccess, emojiReactFail, unEmojiReactRequest, unEmojiReactSuccess, unEmojiReactFail, };