2022-01-10 14:01:24 -08:00
|
|
|
import { isLoggedIn } from 'soapbox/utils/auth';
|
|
|
|
import { getFeatures } from 'soapbox/utils/features';
|
|
|
|
import { shouldHaveCard } from 'soapbox/utils/status';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2020-03-27 13:59:38 -07:00
|
|
|
import api from '../api';
|
2022-01-10 14:25:06 -08:00
|
|
|
|
2021-07-15 10:28:18 -07:00
|
|
|
import { importFetchedStatus, importFetchedStatuses } from './importer';
|
2020-04-14 13:45:38 -07:00
|
|
|
import { openModal } from './modal';
|
2022-01-10 14:17:52 -08:00
|
|
|
import { deleteFromTimelines } from './timelines';
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-06-27 12:17:37 -07:00
|
|
|
export const STATUS_CREATE_REQUEST = 'STATUS_CREATE_REQUEST';
|
|
|
|
export const STATUS_CREATE_SUCCESS = 'STATUS_CREATE_SUCCESS';
|
|
|
|
export const STATUS_CREATE_FAIL = 'STATUS_CREATE_FAIL';
|
|
|
|
|
2020-03-27 13:59:38 -07:00
|
|
|
export const STATUS_FETCH_REQUEST = 'STATUS_FETCH_REQUEST';
|
|
|
|
export const STATUS_FETCH_SUCCESS = 'STATUS_FETCH_SUCCESS';
|
|
|
|
export const STATUS_FETCH_FAIL = 'STATUS_FETCH_FAIL';
|
|
|
|
|
|
|
|
export const STATUS_DELETE_REQUEST = 'STATUS_DELETE_REQUEST';
|
|
|
|
export const STATUS_DELETE_SUCCESS = 'STATUS_DELETE_SUCCESS';
|
|
|
|
export const STATUS_DELETE_FAIL = 'STATUS_DELETE_FAIL';
|
|
|
|
|
|
|
|
export const CONTEXT_FETCH_REQUEST = 'CONTEXT_FETCH_REQUEST';
|
|
|
|
export const CONTEXT_FETCH_SUCCESS = 'CONTEXT_FETCH_SUCCESS';
|
|
|
|
export const CONTEXT_FETCH_FAIL = 'CONTEXT_FETCH_FAIL';
|
|
|
|
|
|
|
|
export const STATUS_MUTE_REQUEST = 'STATUS_MUTE_REQUEST';
|
|
|
|
export const STATUS_MUTE_SUCCESS = 'STATUS_MUTE_SUCCESS';
|
|
|
|
export const STATUS_MUTE_FAIL = 'STATUS_MUTE_FAIL';
|
|
|
|
|
|
|
|
export const STATUS_UNMUTE_REQUEST = 'STATUS_UNMUTE_REQUEST';
|
|
|
|
export const STATUS_UNMUTE_SUCCESS = 'STATUS_UNMUTE_SUCCESS';
|
|
|
|
export const STATUS_UNMUTE_FAIL = 'STATUS_UNMUTE_FAIL';
|
|
|
|
|
|
|
|
export const STATUS_REVEAL = 'STATUS_REVEAL';
|
|
|
|
export const STATUS_HIDE = 'STATUS_HIDE';
|
|
|
|
|
|
|
|
export const REDRAFT = 'REDRAFT';
|
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
const statusExists = (getState, statusId) => {
|
|
|
|
return getState().getIn(['statuses', statusId], null) !== null;
|
|
|
|
};
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-06-27 12:17:37 -07:00
|
|
|
export function createStatus(params, idempotencyKey) {
|
|
|
|
return (dispatch, getState) => {
|
|
|
|
dispatch({ type: STATUS_CREATE_REQUEST, params, idempotencyKey });
|
|
|
|
|
|
|
|
return api(getState).post('/api/v1/statuses', params, {
|
|
|
|
headers: { 'Idempotency-Key': idempotencyKey },
|
|
|
|
}).then(({ data: status }) => {
|
2021-11-15 17:08:27 -08:00
|
|
|
// The backend might still be processing the rich media attachment
|
|
|
|
if (!status.card && shouldHaveCard(status)) {
|
|
|
|
status.expectsCard = true;
|
|
|
|
}
|
|
|
|
|
2021-10-09 19:12:21 -07:00
|
|
|
dispatch(importFetchedStatus(status, idempotencyKey));
|
2021-06-27 12:17:37 -07:00
|
|
|
dispatch({ type: STATUS_CREATE_SUCCESS, status, params, idempotencyKey });
|
2021-11-15 17:08:27 -08:00
|
|
|
|
|
|
|
// 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 && 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);
|
|
|
|
}
|
|
|
|
|
2021-06-27 12:17:37 -07:00
|
|
|
return status;
|
|
|
|
}).catch(error => {
|
|
|
|
dispatch({ type: STATUS_CREATE_FAIL, error, params, idempotencyKey });
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2021-06-27 12:17:37 -07:00
|
|
|
|
2020-03-27 13:59:38 -07:00
|
|
|
export function fetchStatus(id) {
|
|
|
|
return (dispatch, getState) => {
|
2021-11-04 11:16:28 -07:00
|
|
|
const skipLoading = statusExists(getState, id);
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_FETCH_REQUEST, id, skipLoading });
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
return api(getState).get(`/api/v1/statuses/${id}`).then(({ data: status }) => {
|
|
|
|
dispatch(importFetchedStatus(status));
|
|
|
|
dispatch({ type: STATUS_FETCH_SUCCESS, status, skipLoading });
|
|
|
|
return status;
|
2021-07-15 10:28:18 -07:00
|
|
|
}).catch(error => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_FETCH_FAIL, id, error, skipLoading, skipAlert: true });
|
2020-03-27 13:59:38 -07:00
|
|
|
});
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function redraft(status, raw_text) {
|
2022-01-04 12:06:08 -08:00
|
|
|
return (dispatch, getState) => {
|
|
|
|
const state = getState();
|
|
|
|
const instance = state.get('instance');
|
|
|
|
const { explicitAddressing } = getFeatures(instance);
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: REDRAFT,
|
|
|
|
status,
|
|
|
|
raw_text,
|
|
|
|
explicitAddressing,
|
|
|
|
});
|
2020-03-27 13:59:38 -07:00
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function deleteStatus(id, routerHistory, withRedraft = false) {
|
|
|
|
return (dispatch, getState) => {
|
2021-03-24 09:44:51 -07:00
|
|
|
if (!isLoggedIn(getState)) return;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
let status = getState().getIn(['statuses', id]);
|
|
|
|
|
|
|
|
if (status.get('poll')) {
|
|
|
|
status = status.set('poll', getState().getIn(['polls', status.get('poll')]));
|
|
|
|
}
|
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_DELETE_REQUEST, id });
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
api(getState).delete(`/api/v1/statuses/${id}`).then(response => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_DELETE_SUCCESS, id });
|
2020-03-27 13:59:38 -07:00
|
|
|
dispatch(deleteFromTimelines(id));
|
|
|
|
|
|
|
|
if (withRedraft) {
|
|
|
|
dispatch(redraft(status, response.data.text));
|
|
|
|
dispatch(openModal('COMPOSE'));
|
|
|
|
}
|
|
|
|
}).catch(error => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_DELETE_FAIL, id, error });
|
2020-03-27 13:59:38 -07:00
|
|
|
});
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function fetchContext(id) {
|
|
|
|
return (dispatch, getState) => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: CONTEXT_FETCH_REQUEST, id });
|
|
|
|
|
|
|
|
return api(getState).get(`/api/v1/statuses/${id}/context`).then(({ data: context }) => {
|
|
|
|
const { ancestors, descendants } = context;
|
|
|
|
const statuses = ancestors.concat(descendants);
|
|
|
|
dispatch(importFetchedStatuses(statuses));
|
|
|
|
dispatch({ type: CONTEXT_FETCH_SUCCESS, id, ancestors, descendants });
|
|
|
|
return context;
|
2020-03-27 13:59:38 -07:00
|
|
|
}).catch(error => {
|
|
|
|
if (error.response && error.response.status === 404) {
|
|
|
|
dispatch(deleteFromTimelines(id));
|
|
|
|
}
|
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: CONTEXT_FETCH_FAIL, id, error, skipAlert: true });
|
2020-03-27 13:59:38 -07:00
|
|
|
});
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
export function fetchStatusWithContext(id) {
|
|
|
|
return (dispatch, getState) => {
|
|
|
|
return Promise.all([
|
|
|
|
dispatch(fetchContext(id)),
|
|
|
|
dispatch(fetchStatus(id)),
|
|
|
|
]);
|
2020-03-27 13:59:38 -07:00
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function muteStatus(id) {
|
|
|
|
return (dispatch, getState) => {
|
2021-03-24 09:44:51 -07:00
|
|
|
if (!isLoggedIn(getState)) return;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_MUTE_REQUEST, id });
|
2020-03-27 13:59:38 -07:00
|
|
|
api(getState).post(`/api/v1/statuses/${id}/mute`).then(() => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_MUTE_SUCCESS, id });
|
2020-03-27 13:59:38 -07:00
|
|
|
}).catch(error => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_MUTE_FAIL, id, error });
|
2020-03-27 13:59:38 -07:00
|
|
|
});
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function unmuteStatus(id) {
|
|
|
|
return (dispatch, getState) => {
|
2021-03-24 09:44:51 -07:00
|
|
|
if (!isLoggedIn(getState)) return;
|
2020-03-27 13:59:38 -07:00
|
|
|
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_UNMUTE_REQUEST, id });
|
2020-03-27 13:59:38 -07:00
|
|
|
api(getState).post(`/api/v1/statuses/${id}/unmute`).then(() => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_UNMUTE_SUCCESS, id });
|
2020-03-27 13:59:38 -07:00
|
|
|
}).catch(error => {
|
2021-11-04 11:16:28 -07:00
|
|
|
dispatch({ type: STATUS_UNMUTE_FAIL, id, error });
|
2020-03-27 13:59:38 -07:00
|
|
|
});
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function hideStatus(ids) {
|
|
|
|
if (!Array.isArray(ids)) {
|
|
|
|
ids = [ids];
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: STATUS_HIDE,
|
|
|
|
ids,
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|
2020-03-27 13:59:38 -07:00
|
|
|
|
|
|
|
export function revealStatus(ids) {
|
|
|
|
if (!Array.isArray(ids)) {
|
|
|
|
ids = [ids];
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
type: STATUS_REVEAL,
|
|
|
|
ids,
|
|
|
|
};
|
2021-08-03 12:22:51 -07:00
|
|
|
}
|