bigbuffet-rw/app/soapbox/actions/emoji-reacts.ts
2023-06-28 16:27:25 -05:00

191 lines
5.5 KiB
TypeScript

import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
import { isLoggedIn } from 'soapbox/utils/auth';
import api from '../api';
import { importFetchedAccounts, importFetchedStatus } from './importer';
import { favourite, unfavourite } from './interactions';
import type { AxiosError } from 'axios';
import type { AppDispatch, RootState } from 'soapbox/store';
import type { APIEntity, 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 EMOJI_REACTS_FETCH_REQUEST = 'EMOJI_REACTS_FETCH_REQUEST';
const EMOJI_REACTS_FETCH_SUCCESS = 'EMOJI_REACTS_FETCH_SUCCESS';
const EMOJI_REACTS_FETCH_FAIL = 'EMOJI_REACTS_FETCH_FAIL';
const noOp = () => () => new Promise(f => f(undefined));
const simpleEmojiReact = (status: Status, emoji: string, custom?: string) =>
(dispatch: AppDispatch) => {
const emojiReacts: ImmutableList<ImmutableMap<string, any>> = status.pleroma.get('emoji_reactions') || ImmutableList();
if (emoji === '👍' && status.favourited) return dispatch(unfavourite(status));
const undo = emojiReacts.filter(e => e.get('me') === true && e.get('name') === emoji).count() > 0;
if (undo) return dispatch(unEmojiReact(status, emoji));
return Promise.all([
...emojiReacts
.filter((emojiReact) => emojiReact.get('me') === true)
.map(emojiReact => dispatch(unEmojiReact(status, emojiReact.get('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 fetchEmojiReacts = (id: string, emoji: string) =>
(dispatch: AppDispatch, getState: () => RootState) => {
if (!isLoggedIn(getState)) return dispatch(noOp());
dispatch(fetchEmojiReactsRequest(id, emoji));
const url = emoji
? `/api/v1/pleroma/statuses/${id}/reactions/${emoji}`
: `/api/v1/pleroma/statuses/${id}/reactions`;
return api(getState).get(url).then(response => {
response.data.forEach((emojiReact: APIEntity) => {
dispatch(importFetchedAccounts(emojiReact.accounts));
});
dispatch(fetchEmojiReactsSuccess(id, response.data));
}).catch(error => {
dispatch(fetchEmojiReactsFail(id, error));
});
};
const emojiReact = (status: Status, emoji: string, custom?: string) =>
(dispatch: AppDispatch, getState: () => RootState) => {
if (!isLoggedIn(getState)) return dispatch(noOp());
dispatch(emojiReactRequest(status, emoji, custom));
return api(getState)
.put(`/api/v1/pleroma/statuses/${status.id}/reactions/${emoji}`)
.then(function(response) {
dispatch(importFetchedStatus(response.data));
dispatch(emojiReactSuccess(status, emoji));
}).catch(function(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 api(getState)
.delete(`/api/v1/pleroma/statuses/${status.id}/reactions/${emoji}`)
.then(response => {
dispatch(importFetchedStatus(response.data));
dispatch(unEmojiReactSuccess(status, emoji));
}).catch(error => {
dispatch(unEmojiReactFail(status, emoji, error));
});
};
const fetchEmojiReactsRequest = (id: string, emoji: string) => ({
type: EMOJI_REACTS_FETCH_REQUEST,
id,
emoji,
});
const fetchEmojiReactsSuccess = (id: string, emojiReacts: APIEntity[]) => ({
type: EMOJI_REACTS_FETCH_SUCCESS,
id,
emojiReacts,
});
const fetchEmojiReactsFail = (id: string, error: AxiosError) => ({
type: EMOJI_REACTS_FETCH_FAIL,
id,
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: AxiosError) => ({
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: AxiosError) => ({
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,
EMOJI_REACTS_FETCH_REQUEST,
EMOJI_REACTS_FETCH_SUCCESS,
EMOJI_REACTS_FETCH_FAIL,
simpleEmojiReact,
fetchEmojiReacts,
emojiReact,
unEmojiReact,
fetchEmojiReactsRequest,
fetchEmojiReactsSuccess,
fetchEmojiReactsFail,
emojiReactRequest,
emojiReactSuccess,
emojiReactFail,
unEmojiReactRequest,
unEmojiReactSuccess,
unEmojiReactFail,
};