384 lines
12 KiB
TypeScript
384 lines
12 KiB
TypeScript
import { isLoggedIn } from 'soapbox/utils/auth';
|
|
import { getFeatures } from 'soapbox/utils/features';
|
|
import { shouldHaveCard } from 'soapbox/utils/status';
|
|
|
|
import api, { getNextLink } from '../api';
|
|
|
|
import { setComposeToStatus } from './compose';
|
|
import { importFetchedStatus, importFetchedStatuses } from './importer';
|
|
import { openModal } from './modals';
|
|
import { deleteFromTimelines } from './timelines';
|
|
|
|
import type { AppDispatch, RootState } from 'soapbox/store';
|
|
import type { APIEntity, Status } from 'soapbox/types/entities';
|
|
|
|
const STATUS_CREATE_REQUEST = 'STATUS_CREATE_REQUEST';
|
|
const STATUS_CREATE_SUCCESS = 'STATUS_CREATE_SUCCESS';
|
|
const STATUS_CREATE_FAIL = 'STATUS_CREATE_FAIL';
|
|
|
|
const STATUS_FETCH_SOURCE_REQUEST = 'STATUS_FETCH_SOURCE_REQUEST';
|
|
const STATUS_FETCH_SOURCE_SUCCESS = 'STATUS_FETCH_SOURCE_SUCCESS';
|
|
const STATUS_FETCH_SOURCE_FAIL = 'STATUS_FETCH_SOURCE_FAIL';
|
|
|
|
const STATUS_FETCH_REQUEST = 'STATUS_FETCH_REQUEST';
|
|
const STATUS_FETCH_SUCCESS = 'STATUS_FETCH_SUCCESS';
|
|
const STATUS_FETCH_FAIL = 'STATUS_FETCH_FAIL';
|
|
|
|
const STATUS_DELETE_REQUEST = 'STATUS_DELETE_REQUEST';
|
|
const STATUS_DELETE_SUCCESS = 'STATUS_DELETE_SUCCESS';
|
|
const STATUS_DELETE_FAIL = 'STATUS_DELETE_FAIL';
|
|
|
|
const CONTEXT_FETCH_REQUEST = 'CONTEXT_FETCH_REQUEST';
|
|
const CONTEXT_FETCH_SUCCESS = 'CONTEXT_FETCH_SUCCESS';
|
|
const CONTEXT_FETCH_FAIL = 'CONTEXT_FETCH_FAIL';
|
|
|
|
const STATUS_MUTE_REQUEST = 'STATUS_MUTE_REQUEST';
|
|
const STATUS_MUTE_SUCCESS = 'STATUS_MUTE_SUCCESS';
|
|
const STATUS_MUTE_FAIL = 'STATUS_MUTE_FAIL';
|
|
|
|
const STATUS_UNMUTE_REQUEST = 'STATUS_UNMUTE_REQUEST';
|
|
const STATUS_UNMUTE_SUCCESS = 'STATUS_UNMUTE_SUCCESS';
|
|
const STATUS_UNMUTE_FAIL = 'STATUS_UNMUTE_FAIL';
|
|
|
|
const STATUS_REVEAL = 'STATUS_REVEAL';
|
|
const STATUS_HIDE = 'STATUS_HIDE';
|
|
|
|
const STATUS_TRANSLATE_REQUEST = 'STATUS_TRANSLATE_REQUEST';
|
|
const STATUS_TRANSLATE_SUCCESS = 'STATUS_TRANSLATE_SUCCESS';
|
|
const STATUS_TRANSLATE_FAIL = 'STATUS_TRANSLATE_FAIL';
|
|
const STATUS_TRANSLATE_UNDO = 'STATUS_TRANSLATE_UNDO';
|
|
|
|
const statusExists = (getState: () => RootState, statusId: string) => {
|
|
return (getState().statuses.get(statusId) || null) !== null;
|
|
};
|
|
|
|
const createStatus = (params: Record<string, any>, idempotencyKey: string, statusId: string | null) => {
|
|
return (dispatch: AppDispatch, getState: () => RootState) => {
|
|
dispatch({ type: STATUS_CREATE_REQUEST, params, idempotencyKey, editing: !!statusId });
|
|
|
|
return api(getState).request({
|
|
url: statusId === null ? '/api/v1/statuses' : `/api/v1/statuses/${statusId}`,
|
|
method: statusId === null ? 'post' : 'put',
|
|
data: params,
|
|
headers: { 'Idempotency-Key': idempotencyKey },
|
|
}).then(({ data: status }) => {
|
|
// The backend might still be processing the rich media attachment
|
|
if (!status.card && shouldHaveCard(status)) {
|
|
status.expectsCard = true;
|
|
}
|
|
|
|
dispatch(importFetchedStatus(status, idempotencyKey));
|
|
dispatch({ type: STATUS_CREATE_SUCCESS, status, params, idempotencyKey, editing: !!statusId });
|
|
|
|
// Poll the backend for the updated card
|
|
if (status.expectsCard) {
|
|
const delay = 1000;
|
|
|
|
const poll = (retries = 5) => {
|
|
api(getState).get(`/api/v1/statuses/${status.id}`).then(response => {
|
|
if (response.data?.card) {
|
|
dispatch(importFetchedStatus(response.data));
|
|
} else if (retries > 0 && response.status === 200) {
|
|
setTimeout(() => poll(retries - 1), delay);
|
|
}
|
|
}).catch(console.error);
|
|
};
|
|
|
|
setTimeout(() => poll(), delay);
|
|
}
|
|
|
|
return status;
|
|
}).catch(error => {
|
|
dispatch({ type: STATUS_CREATE_FAIL, error, params, idempotencyKey, editing: !!statusId });
|
|
throw error;
|
|
});
|
|
};
|
|
};
|
|
|
|
const editStatus = (id: string) => (dispatch: AppDispatch, getState: () => RootState) => {
|
|
let status = getState().statuses.get(id)!;
|
|
|
|
if (status.poll) {
|
|
status = status.set('poll', getState().polls.get(status.poll) as any);
|
|
}
|
|
|
|
dispatch({ type: STATUS_FETCH_SOURCE_REQUEST });
|
|
|
|
api(getState).get(`/api/v1/statuses/${id}/source`).then(response => {
|
|
dispatch({ type: STATUS_FETCH_SOURCE_SUCCESS });
|
|
dispatch(setComposeToStatus(status, response.data.text, response.data.spoiler_text, response.data.content_type, false));
|
|
dispatch(openModal('COMPOSE'));
|
|
}).catch(error => {
|
|
dispatch({ type: STATUS_FETCH_SOURCE_FAIL, error });
|
|
|
|
});
|
|
};
|
|
|
|
const fetchStatus = (id: string) => {
|
|
return (dispatch: AppDispatch, getState: () => RootState) => {
|
|
const skipLoading = statusExists(getState, id);
|
|
|
|
dispatch({ type: STATUS_FETCH_REQUEST, id, skipLoading });
|
|
|
|
return api(getState).get(`/api/v1/statuses/${id}`).then(({ data: status }) => {
|
|
dispatch(importFetchedStatus(status));
|
|
dispatch({ type: STATUS_FETCH_SUCCESS, status, skipLoading });
|
|
return status;
|
|
}).catch(error => {
|
|
dispatch({ type: STATUS_FETCH_FAIL, id, error, skipLoading, skipAlert: true });
|
|
});
|
|
};
|
|
};
|
|
|
|
const deleteStatus = (id: string, withRedraft = false) => {
|
|
return (dispatch: AppDispatch, getState: () => RootState) => {
|
|
if (!isLoggedIn(getState)) return null;
|
|
|
|
let status = getState().statuses.get(id)!;
|
|
|
|
if (status.poll) {
|
|
status = status.set('poll', getState().polls.get(status.poll) as any);
|
|
}
|
|
|
|
dispatch({ type: STATUS_DELETE_REQUEST, params: status });
|
|
|
|
return api(getState)
|
|
.delete(`/api/v1/statuses/${id}`)
|
|
.then(response => {
|
|
dispatch({ type: STATUS_DELETE_SUCCESS, id });
|
|
dispatch(deleteFromTimelines(id));
|
|
|
|
if (withRedraft) {
|
|
dispatch(setComposeToStatus(status, response.data.text, response.data.spoiler_text, response.data.pleroma?.content_type, withRedraft));
|
|
dispatch(openModal('COMPOSE'));
|
|
}
|
|
})
|
|
.catch(error => {
|
|
dispatch({ type: STATUS_DELETE_FAIL, params: status, error });
|
|
});
|
|
};
|
|
};
|
|
|
|
const updateStatus = (status: APIEntity) => (dispatch: AppDispatch) =>
|
|
dispatch(importFetchedStatus(status));
|
|
|
|
const fetchContext = (id: string) =>
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
dispatch({ type: CONTEXT_FETCH_REQUEST, id });
|
|
|
|
return api(getState).get(`/api/v1/statuses/${id}/context`).then(({ data: context }) => {
|
|
if (Array.isArray(context)) {
|
|
// Mitra: returns a list of statuses
|
|
dispatch(importFetchedStatuses(context));
|
|
} else if (typeof context === 'object') {
|
|
// Standard Mastodon API returns a map with `ancestors` and `descendants`
|
|
const { ancestors, descendants } = context;
|
|
const statuses = ancestors.concat(descendants);
|
|
dispatch(importFetchedStatuses(statuses));
|
|
dispatch({ type: CONTEXT_FETCH_SUCCESS, id, ancestors, descendants });
|
|
} else {
|
|
throw context;
|
|
}
|
|
return context;
|
|
}).catch(error => {
|
|
if (error.response?.status === 404) {
|
|
dispatch(deleteFromTimelines(id));
|
|
}
|
|
|
|
dispatch({ type: CONTEXT_FETCH_FAIL, id, error, skipAlert: true });
|
|
});
|
|
};
|
|
|
|
const fetchNext = (statusId: string, next: string) =>
|
|
async(dispatch: AppDispatch, getState: () => RootState) => {
|
|
const response = await api(getState).get(next);
|
|
dispatch(importFetchedStatuses(response.data));
|
|
|
|
dispatch({
|
|
type: CONTEXT_FETCH_SUCCESS,
|
|
id: statusId,
|
|
ancestors: [],
|
|
descendants: response.data,
|
|
});
|
|
|
|
return { next: getNextLink(response) };
|
|
};
|
|
|
|
const fetchAncestors = (id: string) =>
|
|
async(dispatch: AppDispatch, getState: () => RootState) => {
|
|
const response = await api(getState).get(`/api/v1/statuses/${id}/context/ancestors`);
|
|
dispatch(importFetchedStatuses(response.data));
|
|
return response;
|
|
};
|
|
|
|
const fetchDescendants = (id: string) =>
|
|
async(dispatch: AppDispatch, getState: () => RootState) => {
|
|
const response = await api(getState).get(`/api/v1/statuses/${id}/context/descendants`);
|
|
dispatch(importFetchedStatuses(response.data));
|
|
return response;
|
|
};
|
|
|
|
const fetchStatusWithContext = (id: string) =>
|
|
async(dispatch: AppDispatch, getState: () => RootState) => {
|
|
const features = getFeatures(getState().instance);
|
|
|
|
if (features.paginatedContext) {
|
|
await dispatch(fetchStatus(id));
|
|
const responses = await Promise.all([
|
|
dispatch(fetchAncestors(id)),
|
|
dispatch(fetchDescendants(id)),
|
|
]);
|
|
|
|
dispatch({
|
|
type: CONTEXT_FETCH_SUCCESS,
|
|
id,
|
|
ancestors: responses[0].data,
|
|
descendants: responses[1].data,
|
|
});
|
|
|
|
const next = getNextLink(responses[1]);
|
|
return { next };
|
|
} else {
|
|
await Promise.all([
|
|
dispatch(fetchContext(id)),
|
|
dispatch(fetchStatus(id)),
|
|
]);
|
|
return { next: undefined };
|
|
}
|
|
};
|
|
|
|
const muteStatus = (id: string) =>
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
if (!isLoggedIn(getState)) return;
|
|
|
|
dispatch({ type: STATUS_MUTE_REQUEST, id });
|
|
api(getState).post(`/api/v1/statuses/${id}/mute`).then(() => {
|
|
dispatch({ type: STATUS_MUTE_SUCCESS, id });
|
|
}).catch(error => {
|
|
dispatch({ type: STATUS_MUTE_FAIL, id, error });
|
|
});
|
|
};
|
|
|
|
const unmuteStatus = (id: string) =>
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
if (!isLoggedIn(getState)) return;
|
|
|
|
dispatch({ type: STATUS_UNMUTE_REQUEST, id });
|
|
api(getState).post(`/api/v1/statuses/${id}/unmute`).then(() => {
|
|
dispatch({ type: STATUS_UNMUTE_SUCCESS, id });
|
|
}).catch(error => {
|
|
dispatch({ type: STATUS_UNMUTE_FAIL, id, error });
|
|
});
|
|
};
|
|
|
|
const toggleMuteStatus = (status: Status) =>
|
|
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
if (status.muted) {
|
|
dispatch(unmuteStatus(status.id));
|
|
} else {
|
|
dispatch(muteStatus(status.id));
|
|
}
|
|
};
|
|
|
|
const hideStatus = (ids: string[] | string) => {
|
|
if (!Array.isArray(ids)) {
|
|
ids = [ids];
|
|
}
|
|
|
|
return {
|
|
type: STATUS_HIDE,
|
|
ids,
|
|
};
|
|
};
|
|
|
|
const revealStatus = (ids: string[] | string) => {
|
|
if (!Array.isArray(ids)) {
|
|
ids = [ids];
|
|
}
|
|
|
|
return {
|
|
type: STATUS_REVEAL,
|
|
ids,
|
|
};
|
|
};
|
|
|
|
const toggleStatusHidden = (status: Status) => {
|
|
if (status.hidden) {
|
|
return revealStatus(status.id);
|
|
} else {
|
|
return hideStatus(status.id);
|
|
}
|
|
};
|
|
|
|
const translateStatus = (id: string, targetLanguage?: string) => (dispatch: AppDispatch, getState: () => RootState) => {
|
|
dispatch({ type: STATUS_TRANSLATE_REQUEST, id });
|
|
|
|
api(getState).post(`/api/v1/statuses/${id}/translate`, {
|
|
target_language: targetLanguage,
|
|
}).then(response => {
|
|
dispatch({
|
|
type: STATUS_TRANSLATE_SUCCESS,
|
|
id,
|
|
translation: response.data,
|
|
});
|
|
}).catch(error => {
|
|
dispatch({
|
|
type: STATUS_TRANSLATE_FAIL,
|
|
id,
|
|
error,
|
|
});
|
|
});
|
|
};
|
|
|
|
const undoStatusTranslation = (id: string) => ({
|
|
type: STATUS_TRANSLATE_UNDO,
|
|
id,
|
|
});
|
|
|
|
export {
|
|
STATUS_CREATE_REQUEST,
|
|
STATUS_CREATE_SUCCESS,
|
|
STATUS_CREATE_FAIL,
|
|
STATUS_FETCH_SOURCE_REQUEST,
|
|
STATUS_FETCH_SOURCE_SUCCESS,
|
|
STATUS_FETCH_SOURCE_FAIL,
|
|
STATUS_FETCH_REQUEST,
|
|
STATUS_FETCH_SUCCESS,
|
|
STATUS_FETCH_FAIL,
|
|
STATUS_DELETE_REQUEST,
|
|
STATUS_DELETE_SUCCESS,
|
|
STATUS_DELETE_FAIL,
|
|
CONTEXT_FETCH_REQUEST,
|
|
CONTEXT_FETCH_SUCCESS,
|
|
CONTEXT_FETCH_FAIL,
|
|
STATUS_MUTE_REQUEST,
|
|
STATUS_MUTE_SUCCESS,
|
|
STATUS_MUTE_FAIL,
|
|
STATUS_UNMUTE_REQUEST,
|
|
STATUS_UNMUTE_SUCCESS,
|
|
STATUS_UNMUTE_FAIL,
|
|
STATUS_REVEAL,
|
|
STATUS_HIDE,
|
|
STATUS_TRANSLATE_REQUEST,
|
|
STATUS_TRANSLATE_SUCCESS,
|
|
STATUS_TRANSLATE_FAIL,
|
|
STATUS_TRANSLATE_UNDO,
|
|
createStatus,
|
|
editStatus,
|
|
fetchStatus,
|
|
deleteStatus,
|
|
updateStatus,
|
|
fetchContext,
|
|
fetchNext,
|
|
fetchAncestors,
|
|
fetchDescendants,
|
|
fetchStatusWithContext,
|
|
muteStatus,
|
|
unmuteStatus,
|
|
toggleMuteStatus,
|
|
hideStatus,
|
|
revealStatus,
|
|
toggleStatusHidden,
|
|
translateStatus,
|
|
undoStatusTranslation,
|
|
};
|