Merge remote-tracking branch 'origin/develop' into remove-at-decorators
This commit is contained in:
commit
6855decf88
61 changed files with 302 additions and 1050 deletions
|
@ -5,7 +5,7 @@ import { httpErrorMessages } from 'soapbox/utils/errors';
|
||||||
import type { SnackbarActionSeverity } from './snackbar';
|
import type { SnackbarActionSeverity } from './snackbar';
|
||||||
import type { AnyAction } from '@reduxjs/toolkit';
|
import type { AnyAction } from '@reduxjs/toolkit';
|
||||||
import type { AxiosError } from 'axios';
|
import type { AxiosError } from 'axios';
|
||||||
import type { NotificationObject } from 'soapbox/react-notification';
|
import type { NotificationObject } from 'react-notification';
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
unexpectedTitle: { id: 'alert.unexpected.title', defaultMessage: 'Oops!' },
|
unexpectedTitle: { id: 'alert.unexpected.title', defaultMessage: 'Oops!' },
|
||||||
|
|
|
@ -1,143 +0,0 @@
|
||||||
import { isLoggedIn } from 'soapbox/utils/auth';
|
|
||||||
|
|
||||||
import api from '../api';
|
|
||||||
|
|
||||||
import type { AxiosError } from 'axios';
|
|
||||||
import type { History } from 'history';
|
|
||||||
import type { AppDispatch, RootState } from 'soapbox/store';
|
|
||||||
import type { APIEntity } from 'soapbox/types/entities';
|
|
||||||
|
|
||||||
const GROUP_CREATE_REQUEST = 'GROUP_CREATE_REQUEST';
|
|
||||||
const GROUP_CREATE_SUCCESS = 'GROUP_CREATE_SUCCESS';
|
|
||||||
const GROUP_CREATE_FAIL = 'GROUP_CREATE_FAIL';
|
|
||||||
|
|
||||||
const GROUP_UPDATE_REQUEST = 'GROUP_UPDATE_REQUEST';
|
|
||||||
const GROUP_UPDATE_SUCCESS = 'GROUP_UPDATE_SUCCESS';
|
|
||||||
const GROUP_UPDATE_FAIL = 'GROUP_UPDATE_FAIL';
|
|
||||||
|
|
||||||
const GROUP_EDITOR_VALUE_CHANGE = 'GROUP_EDITOR_VALUE_CHANGE';
|
|
||||||
const GROUP_EDITOR_RESET = 'GROUP_EDITOR_RESET';
|
|
||||||
const GROUP_EDITOR_SETUP = 'GROUP_EDITOR_SETUP';
|
|
||||||
|
|
||||||
const submit = (routerHistory: History) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
const groupId = getState().group_editor.get('groupId') as string;
|
|
||||||
const title = getState().group_editor.get('title') as string;
|
|
||||||
const description = getState().group_editor.get('description') as string;
|
|
||||||
const coverImage = getState().group_editor.get('coverImage') as any;
|
|
||||||
|
|
||||||
if (groupId === null) {
|
|
||||||
dispatch(create(title, description, coverImage, routerHistory));
|
|
||||||
} else {
|
|
||||||
dispatch(update(groupId, title, description, coverImage, routerHistory));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const create = (title: string, description: string, coverImage: File, routerHistory: History) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(createRequest());
|
|
||||||
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('title', title);
|
|
||||||
formData.append('description', description);
|
|
||||||
|
|
||||||
if (coverImage !== null) {
|
|
||||||
formData.append('cover_image', coverImage);
|
|
||||||
}
|
|
||||||
|
|
||||||
api(getState).post('/api/v1/groups', formData, { headers: { 'Content-Type': 'multipart/form-data' } }).then(({ data }) => {
|
|
||||||
dispatch(createSuccess(data));
|
|
||||||
routerHistory.push(`/groups/${data.id}`);
|
|
||||||
}).catch(err => dispatch(createFail(err)));
|
|
||||||
};
|
|
||||||
|
|
||||||
const createRequest = (id?: string) => ({
|
|
||||||
type: GROUP_CREATE_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const createSuccess = (group: APIEntity) => ({
|
|
||||||
type: GROUP_CREATE_SUCCESS,
|
|
||||||
group,
|
|
||||||
});
|
|
||||||
|
|
||||||
const createFail = (error: AxiosError) => ({
|
|
||||||
type: GROUP_CREATE_FAIL,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const update = (groupId: string, title: string, description: string, coverImage: File, routerHistory: History) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(updateRequest(groupId));
|
|
||||||
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('title', title);
|
|
||||||
formData.append('description', description);
|
|
||||||
|
|
||||||
if (coverImage !== null) {
|
|
||||||
formData.append('cover_image', coverImage);
|
|
||||||
}
|
|
||||||
|
|
||||||
api(getState).put(`/api/v1/groups/${groupId}`, formData, { headers: { 'Content-Type': 'multipart/form-data' } }).then(({ data }) => {
|
|
||||||
dispatch(updateSuccess(data));
|
|
||||||
routerHistory.push(`/groups/${data.id}`);
|
|
||||||
}).catch(err => dispatch(updateFail(err)));
|
|
||||||
};
|
|
||||||
|
|
||||||
const updateRequest = (id: string) => ({
|
|
||||||
type: GROUP_UPDATE_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const updateSuccess = (group: APIEntity) => ({
|
|
||||||
type: GROUP_UPDATE_SUCCESS,
|
|
||||||
group,
|
|
||||||
});
|
|
||||||
|
|
||||||
const updateFail = (error: AxiosError) => ({
|
|
||||||
type: GROUP_UPDATE_FAIL,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const changeValue = (field: string, value: string | File) => ({
|
|
||||||
type: GROUP_EDITOR_VALUE_CHANGE,
|
|
||||||
field,
|
|
||||||
value,
|
|
||||||
});
|
|
||||||
|
|
||||||
const reset = () => ({
|
|
||||||
type: GROUP_EDITOR_RESET,
|
|
||||||
});
|
|
||||||
|
|
||||||
const setUp = (group: string) => ({
|
|
||||||
type: GROUP_EDITOR_SETUP,
|
|
||||||
group,
|
|
||||||
});
|
|
||||||
|
|
||||||
export {
|
|
||||||
GROUP_CREATE_REQUEST,
|
|
||||||
GROUP_CREATE_SUCCESS,
|
|
||||||
GROUP_CREATE_FAIL,
|
|
||||||
GROUP_UPDATE_REQUEST,
|
|
||||||
GROUP_UPDATE_SUCCESS,
|
|
||||||
GROUP_UPDATE_FAIL,
|
|
||||||
GROUP_EDITOR_VALUE_CHANGE,
|
|
||||||
GROUP_EDITOR_RESET,
|
|
||||||
GROUP_EDITOR_SETUP,
|
|
||||||
submit,
|
|
||||||
create,
|
|
||||||
createRequest,
|
|
||||||
createSuccess,
|
|
||||||
createFail,
|
|
||||||
update,
|
|
||||||
updateRequest,
|
|
||||||
updateSuccess,
|
|
||||||
updateFail,
|
|
||||||
changeValue,
|
|
||||||
reset,
|
|
||||||
setUp,
|
|
||||||
};
|
|
|
@ -1,550 +0,0 @@
|
||||||
import { AxiosError } from 'axios';
|
|
||||||
|
|
||||||
import { isLoggedIn } from 'soapbox/utils/auth';
|
|
||||||
|
|
||||||
import api, { getLinks } from '../api';
|
|
||||||
|
|
||||||
import { fetchRelationships } from './accounts';
|
|
||||||
import { importFetchedAccounts } from './importer';
|
|
||||||
|
|
||||||
import type { AppDispatch, RootState } from 'soapbox/store';
|
|
||||||
import type { APIEntity } from 'soapbox/types/entities';
|
|
||||||
|
|
||||||
const GROUP_FETCH_REQUEST = 'GROUP_FETCH_REQUEST';
|
|
||||||
const GROUP_FETCH_SUCCESS = 'GROUP_FETCH_SUCCESS';
|
|
||||||
const GROUP_FETCH_FAIL = 'GROUP_FETCH_FAIL';
|
|
||||||
|
|
||||||
const GROUP_RELATIONSHIPS_FETCH_REQUEST = 'GROUP_RELATIONSHIPS_FETCH_REQUEST';
|
|
||||||
const GROUP_RELATIONSHIPS_FETCH_SUCCESS = 'GROUP_RELATIONSHIPS_FETCH_SUCCESS';
|
|
||||||
const GROUP_RELATIONSHIPS_FETCH_FAIL = 'GROUP_RELATIONSHIPS_FETCH_FAIL';
|
|
||||||
|
|
||||||
const GROUPS_FETCH_REQUEST = 'GROUPS_FETCH_REQUEST';
|
|
||||||
const GROUPS_FETCH_SUCCESS = 'GROUPS_FETCH_SUCCESS';
|
|
||||||
const GROUPS_FETCH_FAIL = 'GROUPS_FETCH_FAIL';
|
|
||||||
|
|
||||||
const GROUP_JOIN_REQUEST = 'GROUP_JOIN_REQUEST';
|
|
||||||
const GROUP_JOIN_SUCCESS = 'GROUP_JOIN_SUCCESS';
|
|
||||||
const GROUP_JOIN_FAIL = 'GROUP_JOIN_FAIL';
|
|
||||||
|
|
||||||
const GROUP_LEAVE_REQUEST = 'GROUP_LEAVE_REQUEST';
|
|
||||||
const GROUP_LEAVE_SUCCESS = 'GROUP_LEAVE_SUCCESS';
|
|
||||||
const GROUP_LEAVE_FAIL = 'GROUP_LEAVE_FAIL';
|
|
||||||
|
|
||||||
const GROUP_MEMBERS_FETCH_REQUEST = 'GROUP_MEMBERS_FETCH_REQUEST';
|
|
||||||
const GROUP_MEMBERS_FETCH_SUCCESS = 'GROUP_MEMBERS_FETCH_SUCCESS';
|
|
||||||
const GROUP_MEMBERS_FETCH_FAIL = 'GROUP_MEMBERS_FETCH_FAIL';
|
|
||||||
|
|
||||||
const GROUP_MEMBERS_EXPAND_REQUEST = 'GROUP_MEMBERS_EXPAND_REQUEST';
|
|
||||||
const GROUP_MEMBERS_EXPAND_SUCCESS = 'GROUP_MEMBERS_EXPAND_SUCCESS';
|
|
||||||
const GROUP_MEMBERS_EXPAND_FAIL = 'GROUP_MEMBERS_EXPAND_FAIL';
|
|
||||||
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_FETCH_REQUEST = 'GROUP_REMOVED_ACCOUNTS_FETCH_REQUEST';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_FETCH_SUCCESS = 'GROUP_REMOVED_ACCOUNTS_FETCH_SUCCESS';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_FETCH_FAIL = 'GROUP_REMOVED_ACCOUNTS_FETCH_FAIL';
|
|
||||||
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_EXPAND_REQUEST = 'GROUP_REMOVED_ACCOUNTS_EXPAND_REQUEST';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_EXPAND_SUCCESS = 'GROUP_REMOVED_ACCOUNTS_EXPAND_SUCCESS';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_EXPAND_FAIL = 'GROUP_REMOVED_ACCOUNTS_EXPAND_FAIL';
|
|
||||||
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_REMOVE_REQUEST = 'GROUP_REMOVED_ACCOUNTS_REMOVE_REQUEST';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_REMOVE_SUCCESS = 'GROUP_REMOVED_ACCOUNTS_REMOVE_SUCCESS';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_REMOVE_FAIL = 'GROUP_REMOVED_ACCOUNTS_REMOVE_FAIL';
|
|
||||||
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_CREATE_REQUEST = 'GROUP_REMOVED_ACCOUNTS_CREATE_REQUEST';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_CREATE_SUCCESS = 'GROUP_REMOVED_ACCOUNTS_CREATE_SUCCESS';
|
|
||||||
const GROUP_REMOVED_ACCOUNTS_CREATE_FAIL = 'GROUP_REMOVED_ACCOUNTS_CREATE_FAIL';
|
|
||||||
|
|
||||||
const GROUP_REMOVE_STATUS_REQUEST = 'GROUP_REMOVE_STATUS_REQUEST';
|
|
||||||
const GROUP_REMOVE_STATUS_SUCCESS = 'GROUP_REMOVE_STATUS_SUCCESS';
|
|
||||||
const GROUP_REMOVE_STATUS_FAIL = 'GROUP_REMOVE_STATUS_FAIL';
|
|
||||||
|
|
||||||
const fetchGroup = (id: string) => (dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(fetchGroupRelationships([id]));
|
|
||||||
|
|
||||||
if (getState().groups.get(id)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
dispatch(fetchGroupRequest(id));
|
|
||||||
|
|
||||||
api(getState).get(`/api/v1/groups/${id}`)
|
|
||||||
.then(({ data }) => dispatch(fetchGroupSuccess(data)))
|
|
||||||
.catch(err => dispatch(fetchGroupFail(id, err)));
|
|
||||||
};
|
|
||||||
|
|
||||||
const fetchGroupRequest = (id: string) => ({
|
|
||||||
type: GROUP_FETCH_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupSuccess = (group: APIEntity) => ({
|
|
||||||
type: GROUP_FETCH_SUCCESS,
|
|
||||||
group,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_FETCH_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupRelationships = (groupIds: string[]) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
const loadedRelationships = getState().group_relationships;
|
|
||||||
const newGroupIds = groupIds.filter(id => loadedRelationships.get(id, null) === null);
|
|
||||||
|
|
||||||
if (newGroupIds.length === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
dispatch(fetchGroupRelationshipsRequest(newGroupIds));
|
|
||||||
|
|
||||||
api(getState).get(`/api/v1/groups/${newGroupIds[0]}/relationships?${newGroupIds.map(id => `id[]=${id}`).join('&')}`).then(response => {
|
|
||||||
dispatch(fetchGroupRelationshipsSuccess(response.data));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(fetchGroupRelationshipsFail(error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const fetchGroupRelationshipsRequest = (ids: string[]) => ({
|
|
||||||
type: GROUP_RELATIONSHIPS_FETCH_REQUEST,
|
|
||||||
ids,
|
|
||||||
skipLoading: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupRelationshipsSuccess = (relationships: APIEntity[]) => ({
|
|
||||||
type: GROUP_RELATIONSHIPS_FETCH_SUCCESS,
|
|
||||||
relationships,
|
|
||||||
skipLoading: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupRelationshipsFail = (error: AxiosError) => ({
|
|
||||||
type: GROUP_RELATIONSHIPS_FETCH_FAIL,
|
|
||||||
error,
|
|
||||||
skipLoading: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroups = (tab: string) => (dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(fetchGroupsRequest());
|
|
||||||
|
|
||||||
api(getState).get('/api/v1/groups?tab=' + tab)
|
|
||||||
.then(({ data }) => {
|
|
||||||
dispatch(fetchGroupsSuccess(data, tab));
|
|
||||||
dispatch(fetchGroupRelationships(data.map((item: APIEntity) => item.id)));
|
|
||||||
})
|
|
||||||
.catch(err => dispatch(fetchGroupsFail(err)));
|
|
||||||
};
|
|
||||||
|
|
||||||
const fetchGroupsRequest = () => ({
|
|
||||||
type: GROUPS_FETCH_REQUEST,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupsSuccess = (groups: APIEntity[], tab: string) => ({
|
|
||||||
type: GROUPS_FETCH_SUCCESS,
|
|
||||||
groups,
|
|
||||||
tab,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchGroupsFail = (error: AxiosError) => ({
|
|
||||||
type: GROUPS_FETCH_FAIL,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const joinGroup = (id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(joinGroupRequest(id));
|
|
||||||
|
|
||||||
api(getState).post(`/api/v1/groups/${id}/accounts`).then(response => {
|
|
||||||
dispatch(joinGroupSuccess(response.data));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(joinGroupFail(id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const leaveGroup = (id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(leaveGroupRequest(id));
|
|
||||||
|
|
||||||
api(getState).delete(`/api/v1/groups/${id}/accounts`).then(response => {
|
|
||||||
dispatch(leaveGroupSuccess(response.data));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(leaveGroupFail(id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const joinGroupRequest = (id: string) => ({
|
|
||||||
type: GROUP_JOIN_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const joinGroupSuccess = (relationship: APIEntity) => ({
|
|
||||||
type: GROUP_JOIN_SUCCESS,
|
|
||||||
relationship,
|
|
||||||
});
|
|
||||||
|
|
||||||
const joinGroupFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_JOIN_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const leaveGroupRequest = (id: string) => ({
|
|
||||||
type: GROUP_LEAVE_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const leaveGroupSuccess = (relationship: APIEntity) => ({
|
|
||||||
type: GROUP_LEAVE_SUCCESS,
|
|
||||||
relationship,
|
|
||||||
});
|
|
||||||
|
|
||||||
const leaveGroupFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_LEAVE_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchMembers = (id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(fetchMembersRequest(id));
|
|
||||||
|
|
||||||
api(getState).get(`/api/v1/groups/${id}/accounts`).then(response => {
|
|
||||||
const next = getLinks(response).refs.find(link => link.rel === 'next');
|
|
||||||
|
|
||||||
dispatch(importFetchedAccounts(response.data));
|
|
||||||
dispatch(fetchMembersSuccess(id, response.data, next ? next.uri : null));
|
|
||||||
dispatch(fetchRelationships(response.data.map((item: APIEntity) => item.id)));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(fetchMembersFail(id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const fetchMembersRequest = (id: string) => ({
|
|
||||||
type: GROUP_MEMBERS_FETCH_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchMembersSuccess = (id: string, accounts: APIEntity[], next: string | null) => ({
|
|
||||||
type: GROUP_MEMBERS_FETCH_SUCCESS,
|
|
||||||
id,
|
|
||||||
accounts,
|
|
||||||
next,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchMembersFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_MEMBERS_FETCH_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const expandMembers = (id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
const url = getState().user_lists.groups.get(id)!.next;
|
|
||||||
|
|
||||||
if (url === null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
dispatch(expandMembersRequest(id));
|
|
||||||
|
|
||||||
api(getState).get(url).then(response => {
|
|
||||||
const next = getLinks(response).refs.find(link => link.rel === 'next');
|
|
||||||
|
|
||||||
dispatch(importFetchedAccounts(response.data));
|
|
||||||
dispatch(expandMembersSuccess(id, response.data, next ? next.uri : null));
|
|
||||||
dispatch(fetchRelationships(response.data.map((item: APIEntity) => item.id)));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(expandMembersFail(id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const expandMembersRequest = (id: string) => ({
|
|
||||||
type: GROUP_MEMBERS_EXPAND_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const expandMembersSuccess = (id: string, accounts: APIEntity[], next: string | null) => ({
|
|
||||||
type: GROUP_MEMBERS_EXPAND_SUCCESS,
|
|
||||||
id,
|
|
||||||
accounts,
|
|
||||||
next,
|
|
||||||
});
|
|
||||||
|
|
||||||
const expandMembersFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_MEMBERS_EXPAND_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchRemovedAccounts = (id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(fetchRemovedAccountsRequest(id));
|
|
||||||
|
|
||||||
api(getState).get(`/api/v1/groups/${id}/removed_accounts`).then(response => {
|
|
||||||
const next = getLinks(response).refs.find(link => link.rel === 'next');
|
|
||||||
|
|
||||||
dispatch(importFetchedAccounts(response.data));
|
|
||||||
dispatch(fetchRemovedAccountsSuccess(id, response.data, next ? next.uri : null));
|
|
||||||
dispatch(fetchRelationships(response.data.map((item: APIEntity) => item.id)));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(fetchRemovedAccountsFail(id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const fetchRemovedAccountsRequest = (id: string) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_FETCH_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchRemovedAccountsSuccess = (id: string, accounts: APIEntity[], next: string | null) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_FETCH_SUCCESS,
|
|
||||||
id,
|
|
||||||
accounts,
|
|
||||||
next,
|
|
||||||
});
|
|
||||||
|
|
||||||
const fetchRemovedAccountsFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_FETCH_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const expandRemovedAccounts = (id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
const url = getState().user_lists.groups_removed_accounts.get(id)!.next;
|
|
||||||
|
|
||||||
if (url === null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
dispatch(expandRemovedAccountsRequest(id));
|
|
||||||
|
|
||||||
api(getState).get(url).then(response => {
|
|
||||||
const next = getLinks(response).refs.find(link => link.rel === 'next');
|
|
||||||
|
|
||||||
dispatch(importFetchedAccounts(response.data));
|
|
||||||
dispatch(expandRemovedAccountsSuccess(id, response.data, next ? next.uri : null));
|
|
||||||
dispatch(fetchRelationships(response.data.map((item: APIEntity) => item.id)));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(expandRemovedAccountsFail(id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const expandRemovedAccountsRequest = (id: string) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_EXPAND_REQUEST,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const expandRemovedAccountsSuccess = (id: string, accounts: APIEntity[], next: string | null) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_EXPAND_SUCCESS,
|
|
||||||
id,
|
|
||||||
accounts,
|
|
||||||
next,
|
|
||||||
});
|
|
||||||
|
|
||||||
const expandRemovedAccountsFail = (id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_EXPAND_FAIL,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const removeRemovedAccount = (groupId: string, id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(removeRemovedAccountRequest(groupId, id));
|
|
||||||
|
|
||||||
api(getState).delete(`/api/v1/groups/${groupId}/removed_accounts?account_id=${id}`).then(response => {
|
|
||||||
dispatch(removeRemovedAccountSuccess(groupId, id));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(removeRemovedAccountFail(groupId, id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const removeRemovedAccountRequest = (groupId: string, id: string) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_REMOVE_REQUEST,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const removeRemovedAccountSuccess = (groupId: string, id: string) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_REMOVE_SUCCESS,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const removeRemovedAccountFail = (groupId: string, id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_REMOVE_FAIL,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const createRemovedAccount = (groupId: string, id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(createRemovedAccountRequest(groupId, id));
|
|
||||||
|
|
||||||
api(getState).post(`/api/v1/groups/${groupId}/removed_accounts?account_id=${id}`).then(response => {
|
|
||||||
dispatch(createRemovedAccountSuccess(groupId, id));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(createRemovedAccountFail(groupId, id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const createRemovedAccountRequest = (groupId: string, id: string) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_CREATE_REQUEST,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const createRemovedAccountSuccess = (groupId: string, id: string) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_CREATE_SUCCESS,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const createRemovedAccountFail = (groupId: string, id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_REMOVED_ACCOUNTS_CREATE_FAIL,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
const groupRemoveStatus = (groupId: string, id: string) =>
|
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
|
||||||
if (!isLoggedIn(getState)) return;
|
|
||||||
|
|
||||||
dispatch(groupRemoveStatusRequest(groupId, id));
|
|
||||||
|
|
||||||
api(getState).delete(`/api/v1/groups/${groupId}/statuses/${id}`).then(response => {
|
|
||||||
dispatch(groupRemoveStatusSuccess(groupId, id));
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch(groupRemoveStatusFail(groupId, id, error));
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const groupRemoveStatusRequest = (groupId: string, id: string) => ({
|
|
||||||
type: GROUP_REMOVE_STATUS_REQUEST,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const groupRemoveStatusSuccess = (groupId: string, id: string) => ({
|
|
||||||
type: GROUP_REMOVE_STATUS_SUCCESS,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
});
|
|
||||||
|
|
||||||
const groupRemoveStatusFail = (groupId: string, id: string, error: AxiosError) => ({
|
|
||||||
type: GROUP_REMOVE_STATUS_FAIL,
|
|
||||||
groupId,
|
|
||||||
id,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
|
|
||||||
export {
|
|
||||||
GROUP_FETCH_REQUEST,
|
|
||||||
GROUP_FETCH_SUCCESS,
|
|
||||||
GROUP_FETCH_FAIL,
|
|
||||||
GROUP_RELATIONSHIPS_FETCH_REQUEST,
|
|
||||||
GROUP_RELATIONSHIPS_FETCH_SUCCESS,
|
|
||||||
GROUP_RELATIONSHIPS_FETCH_FAIL,
|
|
||||||
GROUPS_FETCH_REQUEST,
|
|
||||||
GROUPS_FETCH_SUCCESS,
|
|
||||||
GROUPS_FETCH_FAIL,
|
|
||||||
GROUP_JOIN_REQUEST,
|
|
||||||
GROUP_JOIN_SUCCESS,
|
|
||||||
GROUP_JOIN_FAIL,
|
|
||||||
GROUP_LEAVE_REQUEST,
|
|
||||||
GROUP_LEAVE_SUCCESS,
|
|
||||||
GROUP_LEAVE_FAIL,
|
|
||||||
GROUP_MEMBERS_FETCH_REQUEST,
|
|
||||||
GROUP_MEMBERS_FETCH_SUCCESS,
|
|
||||||
GROUP_MEMBERS_FETCH_FAIL,
|
|
||||||
GROUP_MEMBERS_EXPAND_REQUEST,
|
|
||||||
GROUP_MEMBERS_EXPAND_SUCCESS,
|
|
||||||
GROUP_MEMBERS_EXPAND_FAIL,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_FETCH_REQUEST,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_FETCH_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_FETCH_FAIL,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_EXPAND_REQUEST,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_EXPAND_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_EXPAND_FAIL,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_REMOVE_REQUEST,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_REMOVE_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_REMOVE_FAIL,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_CREATE_REQUEST,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_CREATE_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_CREATE_FAIL,
|
|
||||||
GROUP_REMOVE_STATUS_REQUEST,
|
|
||||||
GROUP_REMOVE_STATUS_SUCCESS,
|
|
||||||
GROUP_REMOVE_STATUS_FAIL,
|
|
||||||
fetchGroup,
|
|
||||||
fetchGroupRequest,
|
|
||||||
fetchGroupSuccess,
|
|
||||||
fetchGroupFail,
|
|
||||||
fetchGroupRelationships,
|
|
||||||
fetchGroupRelationshipsRequest,
|
|
||||||
fetchGroupRelationshipsSuccess,
|
|
||||||
fetchGroupRelationshipsFail,
|
|
||||||
fetchGroups,
|
|
||||||
fetchGroupsRequest,
|
|
||||||
fetchGroupsSuccess,
|
|
||||||
fetchGroupsFail,
|
|
||||||
joinGroup,
|
|
||||||
leaveGroup,
|
|
||||||
joinGroupRequest,
|
|
||||||
joinGroupSuccess,
|
|
||||||
joinGroupFail,
|
|
||||||
leaveGroupRequest,
|
|
||||||
leaveGroupSuccess,
|
|
||||||
leaveGroupFail,
|
|
||||||
fetchMembers,
|
|
||||||
fetchMembersRequest,
|
|
||||||
fetchMembersSuccess,
|
|
||||||
fetchMembersFail,
|
|
||||||
expandMembers,
|
|
||||||
expandMembersRequest,
|
|
||||||
expandMembersSuccess,
|
|
||||||
expandMembersFail,
|
|
||||||
fetchRemovedAccounts,
|
|
||||||
fetchRemovedAccountsRequest,
|
|
||||||
fetchRemovedAccountsSuccess,
|
|
||||||
fetchRemovedAccountsFail,
|
|
||||||
expandRemovedAccounts,
|
|
||||||
expandRemovedAccountsRequest,
|
|
||||||
expandRemovedAccountsSuccess,
|
|
||||||
expandRemovedAccountsFail,
|
|
||||||
removeRemovedAccount,
|
|
||||||
removeRemovedAccountRequest,
|
|
||||||
removeRemovedAccountSuccess,
|
|
||||||
removeRemovedAccountFail,
|
|
||||||
createRemovedAccount,
|
|
||||||
createRemovedAccountRequest,
|
|
||||||
createRemovedAccountSuccess,
|
|
||||||
createRemovedAccountFail,
|
|
||||||
groupRemoveStatus,
|
|
||||||
groupRemoveStatusRequest,
|
|
||||||
groupRemoveStatusSuccess,
|
|
||||||
groupRemoveStatusFail,
|
|
||||||
};
|
|
Binary file not shown.
|
@ -8,6 +8,7 @@ import { Text, Stack } from 'soapbox/components/ui';
|
||||||
import { captureException } from 'soapbox/monitoring';
|
import { captureException } from 'soapbox/monitoring';
|
||||||
import KVStore from 'soapbox/storage/kv_store';
|
import KVStore from 'soapbox/storage/kv_store';
|
||||||
import sourceCode from 'soapbox/utils/code';
|
import sourceCode from 'soapbox/utils/code';
|
||||||
|
import { unregisterSw } from 'soapbox/utils/sw';
|
||||||
|
|
||||||
import SiteLogo from './site-logo';
|
import SiteLogo from './site-logo';
|
||||||
|
|
||||||
|
@ -15,16 +16,6 @@ import type { RootState } from 'soapbox/store';
|
||||||
|
|
||||||
const goHome = () => location.href = '/';
|
const goHome = () => location.href = '/';
|
||||||
|
|
||||||
/** Unregister the ServiceWorker */
|
|
||||||
// https://stackoverflow.com/a/49771828/8811886
|
|
||||||
const unregisterSw = async(): Promise<void> => {
|
|
||||||
if (navigator.serviceWorker) {
|
|
||||||
const registrations = await navigator.serviceWorker.getRegistrations();
|
|
||||||
const unregisterAll = registrations.map(r => r.unregister());
|
|
||||||
await Promise.all(unregisterAll);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const mapStateToProps = (state: RootState) => {
|
const mapStateToProps = (state: RootState) => {
|
||||||
const { links, logo } = getSoapboxConfig(state);
|
const { links, logo } = getSoapboxConfig(state);
|
||||||
|
|
||||||
|
|
Binary file not shown.
|
@ -1,15 +1,18 @@
|
||||||
import classNames from 'clsx';
|
import classNames from 'clsx';
|
||||||
import React, { useState } from 'react';
|
import React, { MouseEventHandler, useState } from 'react';
|
||||||
import { defineMessages, useIntl, FormattedMessage, FormattedList } from 'react-intl';
|
import { defineMessages, useIntl } from 'react-intl';
|
||||||
import { useHistory } from 'react-router-dom';
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
import StatusMedia from 'soapbox/components/status-media';
|
import StatusMedia from 'soapbox/components/status-media';
|
||||||
import { Stack, Text } from 'soapbox/components/ui';
|
import { Stack } from 'soapbox/components/ui';
|
||||||
import AccountContainer from 'soapbox/containers/account_container';
|
import AccountContainer from 'soapbox/containers/account_container';
|
||||||
import { useSettings } from 'soapbox/hooks';
|
import { useSettings } from 'soapbox/hooks';
|
||||||
import { defaultMediaVisibility } from 'soapbox/utils/status';
|
import { defaultMediaVisibility } from 'soapbox/utils/status';
|
||||||
|
|
||||||
import OutlineBox from './outline-box';
|
import OutlineBox from './outline-box';
|
||||||
|
import StatusReplyMentions from './status-reply-mentions';
|
||||||
|
import StatusContent from './status_content';
|
||||||
|
import SensitiveContentOverlay from './statuses/sensitive-content-overlay';
|
||||||
|
|
||||||
import type { Account as AccountEntity, Status as StatusEntity } from 'soapbox/types/entities';
|
import type { Account as AccountEntity, Status as StatusEntity } from 'soapbox/types/entities';
|
||||||
|
|
||||||
|
@ -36,7 +39,7 @@ const QuotedStatus: React.FC<IQuotedStatus> = ({ status, onCancel, compose }) =>
|
||||||
|
|
||||||
const [showMedia, setShowMedia] = useState<boolean>(defaultMediaVisibility(status, displayMedia));
|
const [showMedia, setShowMedia] = useState<boolean>(defaultMediaVisibility(status, displayMedia));
|
||||||
|
|
||||||
const handleExpandClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
const handleExpandClick: MouseEventHandler<HTMLDivElement> = (e) => {
|
||||||
if (!status) return;
|
if (!status) return;
|
||||||
const account = status.account as AccountEntity;
|
const account = status.account as AccountEntity;
|
||||||
|
|
||||||
|
@ -57,57 +60,6 @@ const QuotedStatus: React.FC<IQuotedStatus> = ({ status, onCancel, compose }) =>
|
||||||
setShowMedia(!showMedia);
|
setShowMedia(!showMedia);
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderReplyMentions = () => {
|
|
||||||
if (!status?.in_reply_to_id) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const account = status.account as AccountEntity;
|
|
||||||
const to = status.mentions || [];
|
|
||||||
|
|
||||||
if (to.size === 0) {
|
|
||||||
if (status.in_reply_to_account_id === account.id) {
|
|
||||||
return (
|
|
||||||
<div className='reply-mentions'>
|
|
||||||
<FormattedMessage
|
|
||||||
id='reply_mentions.reply'
|
|
||||||
defaultMessage='Replying to {accounts}'
|
|
||||||
values={{
|
|
||||||
accounts: `@${account.username}`,
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return (
|
|
||||||
<div className='reply-mentions'>
|
|
||||||
<FormattedMessage id='reply_mentions.reply_empty' defaultMessage='Replying to post' />
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const accounts = to.slice(0, 2).map(account => <>@{account.username}</>).toArray();
|
|
||||||
|
|
||||||
if (to.size > 2) {
|
|
||||||
accounts.push(
|
|
||||||
<FormattedMessage id='reply_mentions.more' defaultMessage='{count} more' values={{ count: to.size - 2 }} />,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className='reply-mentions'>
|
|
||||||
<FormattedMessage
|
|
||||||
id='reply_mentions.reply'
|
|
||||||
defaultMessage='Replying to {accounts}'
|
|
||||||
values={{
|
|
||||||
accounts: <FormattedList type='conjunction' value={accounts} />,
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!status) {
|
if (!status) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -127,7 +79,7 @@ const QuotedStatus: React.FC<IQuotedStatus> = ({ status, onCancel, compose }) =>
|
||||||
return (
|
return (
|
||||||
<OutlineBox
|
<OutlineBox
|
||||||
data-testid='quoted-status'
|
data-testid='quoted-status'
|
||||||
className={classNames('mt-3 cursor-pointer', {
|
className={classNames('cursor-pointer', {
|
||||||
'hover:bg-gray-100 dark:hover:bg-gray-800': !compose,
|
'hover:bg-gray-100 dark:hover:bg-gray-800': !compose,
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
|
@ -144,20 +96,36 @@ const QuotedStatus: React.FC<IQuotedStatus> = ({ status, onCancel, compose }) =>
|
||||||
withLinkToProfile={!compose}
|
withLinkToProfile={!compose}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{renderReplyMentions()}
|
<StatusReplyMentions status={status} hoverable={false} />
|
||||||
|
|
||||||
<Text
|
<Stack className={classNames('relative', {
|
||||||
className='break-words status__content status__content--quote'
|
'min-h-[220px]': status.hidden,
|
||||||
size='sm'
|
})}
|
||||||
dangerouslySetInnerHTML={{ __html: status.contentHtml }}
|
>
|
||||||
/>
|
{(status.hidden) && (
|
||||||
|
<SensitiveContentOverlay
|
||||||
|
status={status}
|
||||||
|
visible={showMedia}
|
||||||
|
onToggleVisibility={handleToggleMediaVisibility}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
<StatusMedia
|
<Stack space={4}>
|
||||||
status={status}
|
<StatusContent
|
||||||
muted={compose}
|
status={status}
|
||||||
showMedia={showMedia}
|
collapsable
|
||||||
onToggleVisibility={handleToggleMediaVisibility}
|
/>
|
||||||
/>
|
|
||||||
|
{(status.media_attachments.size > 0) && (
|
||||||
|
<StatusMedia
|
||||||
|
status={status}
|
||||||
|
muted={compose}
|
||||||
|
showMedia={showMedia}
|
||||||
|
onToggleVisibility={handleToggleMediaVisibility}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
|
</Stack>
|
||||||
</Stack>
|
</Stack>
|
||||||
</OutlineBox>
|
</OutlineBox>
|
||||||
);
|
);
|
||||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -65,7 +65,6 @@ const Status: React.FC<IStatus> = (props) => {
|
||||||
hidden,
|
hidden,
|
||||||
featured,
|
featured,
|
||||||
unread,
|
unread,
|
||||||
group,
|
|
||||||
hideActionBar,
|
hideActionBar,
|
||||||
variant = 'rounded',
|
variant = 'rounded',
|
||||||
withDismiss,
|
withDismiss,
|
||||||
|
@ -296,8 +295,8 @@ const Status: React.FC<IStatus> = (props) => {
|
||||||
|
|
||||||
const accountAction = props.accountAction || reblogElement;
|
const accountAction = props.accountAction || reblogElement;
|
||||||
|
|
||||||
const inReview = status.visibility === 'self';
|
const inReview = actualStatus.visibility === 'self';
|
||||||
const isSensitive = status.hidden;
|
const isSensitive = actualStatus.hidden;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HotKeys handlers={handlers} data-testid='status'>
|
<HotKeys handlers={handlers} data-testid='status'>
|
||||||
|
@ -349,6 +348,8 @@ const Status: React.FC<IStatus> = (props) => {
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='status__content-wrapper'>
|
<div className='status__content-wrapper'>
|
||||||
|
<StatusReplyMentions status={actualStatus} hoverable={hoverable} />
|
||||||
|
|
||||||
<Stack
|
<Stack
|
||||||
className={
|
className={
|
||||||
classNames('relative', {
|
classNames('relative', {
|
||||||
|
@ -356,40 +357,35 @@ const Status: React.FC<IStatus> = (props) => {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{(inReview || isSensitive) ? (
|
{(inReview || isSensitive) && (
|
||||||
<SensitiveContentOverlay
|
<SensitiveContentOverlay
|
||||||
status={status}
|
status={status}
|
||||||
visible={showMedia}
|
visible={showMedia}
|
||||||
onToggleVisibility={handleToggleMediaVisibility}
|
onToggleVisibility={handleToggleMediaVisibility}
|
||||||
/>
|
/>
|
||||||
) : null}
|
|
||||||
|
|
||||||
{!group && actualStatus.group && (
|
|
||||||
<div className='status__meta'>
|
|
||||||
Posted in <NavLink to={`/groups/${actualStatus.getIn(['group', 'id'])}`}>{String(actualStatus.getIn(['group', 'title']))}</NavLink>
|
|
||||||
</div>
|
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<StatusReplyMentions
|
<Stack space={4}>
|
||||||
status={actualStatus}
|
<StatusContent
|
||||||
hoverable={hoverable}
|
status={actualStatus}
|
||||||
/>
|
onClick={handleClick}
|
||||||
|
collapsable
|
||||||
|
/>
|
||||||
|
|
||||||
<StatusContent
|
{(quote || actualStatus.media_attachments.size > 0) && (
|
||||||
status={actualStatus}
|
<Stack space={4}>
|
||||||
onClick={handleClick}
|
<StatusMedia
|
||||||
collapsable
|
status={actualStatus}
|
||||||
/>
|
muted={muted}
|
||||||
|
onClick={handleClick}
|
||||||
|
showMedia={showMedia}
|
||||||
|
onToggleVisibility={handleToggleMediaVisibility}
|
||||||
|
/>
|
||||||
|
|
||||||
<StatusMedia
|
{quote}
|
||||||
status={actualStatus}
|
</Stack>
|
||||||
muted={muted}
|
)}
|
||||||
onClick={handleClick}
|
</Stack>
|
||||||
showMedia={showMedia}
|
|
||||||
onToggleVisibility={handleToggleMediaVisibility}
|
|
||||||
/>
|
|
||||||
|
|
||||||
{quote}
|
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|
||||||
{!hideActionBar && (
|
{!hideActionBar && (
|
||||||
|
|
|
@ -4,34 +4,22 @@ import { defineMessages, useIntl } from 'react-intl';
|
||||||
// import { connect } from 'react-redux';
|
// import { connect } from 'react-redux';
|
||||||
import { useHistory } from 'react-router-dom';
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
// import { openModal } from 'soapbox/actions/modals';
|
|
||||||
// import { useAppDispatch } from 'soapbox/hooks';
|
|
||||||
|
|
||||||
import { CardHeader, CardTitle } from './ui';
|
import { CardHeader, CardTitle } from './ui';
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
back: { id: 'column_back_button.label', defaultMessage: 'Back' },
|
back: { id: 'column_back_button.label', defaultMessage: 'Back' },
|
||||||
settings: { id: 'column_header.show_settings', defaultMessage: 'Show settings' },
|
|
||||||
});
|
});
|
||||||
|
|
||||||
interface ISubNavigation {
|
interface ISubNavigation {
|
||||||
message: String,
|
message: React.ReactNode,
|
||||||
|
/** @deprecated Unused. */
|
||||||
settings?: React.ComponentType,
|
settings?: React.ComponentType,
|
||||||
}
|
}
|
||||||
|
|
||||||
const SubNavigation: React.FC<ISubNavigation> = ({ message }) => {
|
const SubNavigation: React.FC<ISubNavigation> = ({ message }) => {
|
||||||
const intl = useIntl();
|
const intl = useIntl();
|
||||||
// const dispatch = useAppDispatch();
|
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
|
|
||||||
// const ref = useRef(null);
|
|
||||||
|
|
||||||
// const [scrolled, setScrolled] = useState(false);
|
|
||||||
|
|
||||||
// const onOpenSettings = () => {
|
|
||||||
// dispatch(openModal('COMPONENT', { component: Settings }));
|
|
||||||
// };
|
|
||||||
|
|
||||||
const handleBackClick = () => {
|
const handleBackClick = () => {
|
||||||
if (window.history && window.history.length === 1) {
|
if (window.history && window.history.length === 1) {
|
||||||
history.push('/');
|
history.push('/');
|
||||||
|
@ -40,36 +28,6 @@ const SubNavigation: React.FC<ISubNavigation> = ({ message }) => {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// const handleBackKeyUp = (e) => {
|
|
||||||
// if (e.key === 'Enter') {
|
|
||||||
// handleClick();
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
|
|
||||||
// const handleOpenSettings = () => {
|
|
||||||
// onOpenSettings();
|
|
||||||
// }
|
|
||||||
|
|
||||||
// useEffect(() => {
|
|
||||||
// const handleScroll = throttle(() => {
|
|
||||||
// if (this.node) {
|
|
||||||
// const { offsetTop } = this.node;
|
|
||||||
|
|
||||||
// if (offsetTop > 0) {
|
|
||||||
// setScrolled(true);
|
|
||||||
// } else {
|
|
||||||
// setScrolled(false);
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }, 150, { trailing: true });
|
|
||||||
|
|
||||||
// window.addEventListener('scroll', handleScroll);
|
|
||||||
|
|
||||||
// return () => {
|
|
||||||
// window.removeEventListener('scroll', handleScroll);
|
|
||||||
// };
|
|
||||||
// }, []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<CardHeader
|
<CardHeader
|
||||||
aria-label={intl.formatMessage(messages.back)}
|
aria-label={intl.formatMessage(messages.back)}
|
||||||
|
|
Binary file not shown.
Binary file not shown.
|
@ -10,8 +10,6 @@ import { useAppDispatch, useSettings } from 'soapbox/hooks';
|
||||||
|
|
||||||
import Timeline from '../ui/components/timeline';
|
import Timeline from '../ui/components/timeline';
|
||||||
|
|
||||||
import ColumnSettings from './containers/column_settings_container';
|
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
title: { id: 'column.community', defaultMessage: 'Local timeline' },
|
title: { id: 'column.community', defaultMessage: 'Local timeline' },
|
||||||
});
|
});
|
||||||
|
@ -44,7 +42,10 @@ const CommunityTimeline = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column label={intl.formatMessage(messages.title)} transparent withHeader={false}>
|
<Column label={intl.formatMessage(messages.title)} transparent withHeader={false}>
|
||||||
<SubNavigation message={intl.formatMessage(messages.title)} settings={ColumnSettings} />
|
<div className='px-4 sm:p-0'>
|
||||||
|
<SubNavigation message={intl.formatMessage(messages.title)} />
|
||||||
|
</div>
|
||||||
|
|
||||||
<PullToRefresh onRefresh={handleRefresh}>
|
<PullToRefresh onRefresh={handleRefresh}>
|
||||||
<Timeline
|
<Timeline
|
||||||
scrollKey={`${timelineId}_timeline`}
|
scrollKey={`${timelineId}_timeline`}
|
||||||
|
|
24
app/soapbox/features/developers/components/indicator.tsx
Normal file
24
app/soapbox/features/developers/components/indicator.tsx
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
import classNames from 'clsx';
|
||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
interface IIndicator {
|
||||||
|
state?: 'active' | 'pending' | 'error' | 'inactive',
|
||||||
|
size?: 'sm',
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Indicator dot component. */
|
||||||
|
const Indicator: React.FC<IIndicator> = ({ state = 'inactive', size = 'sm' }) => {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={classNames('rounded-full outline-double', {
|
||||||
|
'w-1.5 h-1.5 shadow-sm': size === 'sm',
|
||||||
|
'bg-green-500 outline-green-400': state === 'active',
|
||||||
|
'bg-yellow-500 outline-yellow-400': state === 'pending',
|
||||||
|
'bg-red-500 outline-red-400': state === 'error',
|
||||||
|
'bg-neutral-500 outline-neutral-400': state === 'inactive',
|
||||||
|
})}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Indicator;
|
|
@ -89,6 +89,14 @@ const Developers: React.FC = () => {
|
||||||
</Text>
|
</Text>
|
||||||
</DashWidget>
|
</DashWidget>
|
||||||
|
|
||||||
|
<DashWidget to='/developers/sw'>
|
||||||
|
<SvgIcon src={require('@tabler/icons/script.svg')} className='text-gray-700 dark:text-gray-600' />
|
||||||
|
|
||||||
|
<Text>
|
||||||
|
<FormattedMessage id='developers.navigation.service_worker_label' defaultMessage='Service Worker' />
|
||||||
|
</Text>
|
||||||
|
</DashWidget>
|
||||||
|
|
||||||
<DashWidget onClick={leaveDevelopers}>
|
<DashWidget onClick={leaveDevelopers}>
|
||||||
<SvgIcon src={require('@tabler/icons/logout.svg')} className='text-gray-700 dark:text-gray-600' />
|
<SvgIcon src={require('@tabler/icons/logout.svg')} className='text-gray-700 dark:text-gray-600' />
|
||||||
|
|
||||||
|
|
140
app/soapbox/features/developers/service-worker-info.tsx
Normal file
140
app/soapbox/features/developers/service-worker-info.tsx
Normal file
|
@ -0,0 +1,140 @@
|
||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
||||||
|
|
||||||
|
import List, { ListItem } from 'soapbox/components/list';
|
||||||
|
import { HStack, Text, Column, FormActions, Button, Stack, Icon } from 'soapbox/components/ui';
|
||||||
|
import { unregisterSw } from 'soapbox/utils/sw';
|
||||||
|
|
||||||
|
import Indicator from './components/indicator';
|
||||||
|
|
||||||
|
const messages = defineMessages({
|
||||||
|
heading: { id: 'column.developers.service_worker', defaultMessage: 'Service Worker' },
|
||||||
|
status: { id: 'sw.status', defaultMessage: 'Status' },
|
||||||
|
url: { id: 'sw.url', defaultMessage: 'Script URL' },
|
||||||
|
});
|
||||||
|
|
||||||
|
/** Hook that returns the active ServiceWorker registration. */
|
||||||
|
const useRegistration = () => {
|
||||||
|
const [isLoading, setLoading] = useState(true);
|
||||||
|
const [registration, setRegistration] = useState<ServiceWorkerRegistration>();
|
||||||
|
|
||||||
|
const isSupported = 'serviceWorker' in navigator;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isSupported) {
|
||||||
|
navigator.serviceWorker.getRegistration()
|
||||||
|
.then(r => {
|
||||||
|
setRegistration(r);
|
||||||
|
setLoading(false);
|
||||||
|
})
|
||||||
|
.catch(() => setLoading(false));
|
||||||
|
} else {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return {
|
||||||
|
isLoading,
|
||||||
|
registration,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
interface IServiceWorkerInfo {
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Mini ServiceWorker debugging component. */
|
||||||
|
const ServiceWorkerInfo: React.FC<IServiceWorkerInfo> = () => {
|
||||||
|
const intl = useIntl();
|
||||||
|
const { isLoading, registration } = useRegistration();
|
||||||
|
|
||||||
|
const url = registration?.active?.scriptURL;
|
||||||
|
|
||||||
|
const getState = () => {
|
||||||
|
if (registration?.waiting) {
|
||||||
|
return 'pending';
|
||||||
|
} else if (registration?.active) {
|
||||||
|
return 'active';
|
||||||
|
} else {
|
||||||
|
return 'inactive';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getMessage = () => {
|
||||||
|
if (isLoading) {
|
||||||
|
return (
|
||||||
|
<FormattedMessage
|
||||||
|
id='sw.state.loading'
|
||||||
|
defaultMessage='Loading…'
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else if (!isLoading && !registration) {
|
||||||
|
return (
|
||||||
|
<FormattedMessage
|
||||||
|
id='sw.state.unavailable'
|
||||||
|
defaultMessage='Unavailable'
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else if (registration?.waiting) {
|
||||||
|
return (
|
||||||
|
<FormattedMessage
|
||||||
|
id='sw.state.waiting'
|
||||||
|
defaultMessage='Waiting'
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else if (registration?.active) {
|
||||||
|
return (
|
||||||
|
<FormattedMessage
|
||||||
|
id='sw.state.active'
|
||||||
|
defaultMessage='Active'
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return (
|
||||||
|
<FormattedMessage
|
||||||
|
id='sw.state.unknown'
|
||||||
|
defaultMessage='Unknown'
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRestart = async() => {
|
||||||
|
await unregisterSw();
|
||||||
|
window.location.reload();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Column label={intl.formatMessage(messages.heading)} backHref='/developers'>
|
||||||
|
<Stack space={4}>
|
||||||
|
<List>
|
||||||
|
<ListItem label={intl.formatMessage(messages.status)}>
|
||||||
|
<HStack alignItems='center' space={2}>
|
||||||
|
<Indicator state={getState()} />
|
||||||
|
<Text size='md' theme='muted'>{getMessage()}</Text>
|
||||||
|
</HStack>
|
||||||
|
</ListItem>
|
||||||
|
|
||||||
|
{url && (
|
||||||
|
<ListItem label={intl.formatMessage(messages.url)}>
|
||||||
|
<a href={url} target='_blank' className='flex space-x-1 items-center truncate'>
|
||||||
|
<span className='truncate'>{url}</span>
|
||||||
|
<Icon
|
||||||
|
className='w-4 h-4'
|
||||||
|
src={require('@tabler/icons/external-link.svg')}
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</ListItem>
|
||||||
|
)}
|
||||||
|
</List>
|
||||||
|
|
||||||
|
<FormActions>
|
||||||
|
<Button theme='tertiary' type='button' onClick={handleRestart}>
|
||||||
|
<FormattedMessage id='sw.restart' defaultMessage='Restart' />
|
||||||
|
</Button>
|
||||||
|
</FormActions>
|
||||||
|
</Stack>
|
||||||
|
</Column>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ServiceWorkerInfo;
|
|
@ -3,10 +3,10 @@ import { FormattedMessage } from 'react-intl';
|
||||||
|
|
||||||
import { connectHashtagStream } from 'soapbox/actions/streaming';
|
import { connectHashtagStream } from 'soapbox/actions/streaming';
|
||||||
import { expandHashtagTimeline, clearTimeline } from 'soapbox/actions/timelines';
|
import { expandHashtagTimeline, clearTimeline } from 'soapbox/actions/timelines';
|
||||||
import ColumnHeader from 'soapbox/components/column_header';
|
import SubNavigation from 'soapbox/components/sub_navigation';
|
||||||
import { Column } from 'soapbox/components/ui';
|
import { Column } from 'soapbox/components/ui';
|
||||||
import Timeline from 'soapbox/features/ui/components/timeline';
|
import Timeline from 'soapbox/features/ui/components/timeline';
|
||||||
import { useAppDispatch, useAppSelector } from 'soapbox/hooks';
|
import { useAppDispatch } from 'soapbox/hooks';
|
||||||
|
|
||||||
import type { Tag as TagEntity } from 'soapbox/types/entities';
|
import type { Tag as TagEntity } from 'soapbox/types/entities';
|
||||||
|
|
||||||
|
@ -27,7 +27,6 @@ export const HashtagTimeline: React.FC<IHashtagTimeline> = ({ params }) => {
|
||||||
const tags = params?.tags || { any: [], all: [], none: [] };
|
const tags = params?.tags || { any: [], all: [], none: [] };
|
||||||
|
|
||||||
const dispatch = useAppDispatch();
|
const dispatch = useAppDispatch();
|
||||||
const hasUnread = useAppSelector<boolean>(state => (state.timelines.getIn([`hashtag:${id}`, 'unread']) as number) > 0);
|
|
||||||
const disconnects = useRef<(() => void)[]>([]);
|
const disconnects = useRef<(() => void)[]>([]);
|
||||||
|
|
||||||
// Mastodon supports displaying results from multiple hashtags.
|
// Mastodon supports displaying results from multiple hashtags.
|
||||||
|
@ -100,7 +99,10 @@ export const HashtagTimeline: React.FC<IHashtagTimeline> = ({ params }) => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column label={`#${id}`} transparent withHeader={false}>
|
<Column label={`#${id}`} transparent withHeader={false}>
|
||||||
<ColumnHeader active={hasUnread} title={title()} />
|
<div className='px-4 pt-4 sm:p-0'>
|
||||||
|
<SubNavigation message={title()} />
|
||||||
|
</div>
|
||||||
|
|
||||||
<Timeline
|
<Timeline
|
||||||
scrollKey='hashtag_timeline'
|
scrollKey='hashtag_timeline'
|
||||||
timelineId={`hashtag:${id}`}
|
timelineId={`hashtag:${id}`}
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -14,8 +14,6 @@ import { useAppDispatch, useAppSelector, useSettings } from 'soapbox/hooks';
|
||||||
import PinnedHostsPicker from '../remote_timeline/components/pinned_hosts_picker';
|
import PinnedHostsPicker from '../remote_timeline/components/pinned_hosts_picker';
|
||||||
import Timeline from '../ui/components/timeline';
|
import Timeline from '../ui/components/timeline';
|
||||||
|
|
||||||
import ColumnSettings from './containers/column_settings_container';
|
|
||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
title: { id: 'column.public', defaultMessage: 'Fediverse timeline' },
|
title: { id: 'column.public', defaultMessage: 'Fediverse timeline' },
|
||||||
dismiss: { id: 'fediverse_tab.explanation_box.dismiss', defaultMessage: 'Don\'t show again' },
|
dismiss: { id: 'fediverse_tab.explanation_box.dismiss', defaultMessage: 'Don\'t show again' },
|
||||||
|
@ -65,8 +63,12 @@ const CommunityTimeline = () => {
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Column label={intl.formatMessage(messages.title)} transparent withHeader={false}>
|
<Column label={intl.formatMessage(messages.title)} transparent withHeader={false}>
|
||||||
<SubNavigation message={intl.formatMessage(messages.title)} settings={ColumnSettings} />
|
<div className='px-4 sm:p-0'>
|
||||||
|
<SubNavigation message={intl.formatMessage(messages.title)} />
|
||||||
|
</div>
|
||||||
|
|
||||||
<PinnedHostsPicker />
|
<PinnedHostsPicker />
|
||||||
|
|
||||||
{showExplanationBox && <div className='mb-4'>
|
{showExplanationBox && <div className='mb-4'>
|
||||||
<Accordion
|
<Accordion
|
||||||
headline={<FormattedMessage id='fediverse_tab.explanation_box.title' defaultMessage='What is the Fediverse?' />}
|
headline={<FormattedMessage id='fediverse_tab.explanation_box.title' defaultMessage='What is the Fediverse?' />}
|
||||||
|
|
|
@ -29,7 +29,6 @@ interface IDetailedStatus {
|
||||||
|
|
||||||
const DetailedStatus: React.FC<IDetailedStatus> = ({
|
const DetailedStatus: React.FC<IDetailedStatus> = ({
|
||||||
status,
|
status,
|
||||||
onToggleHidden,
|
|
||||||
onOpenCompareHistoryModal,
|
onOpenCompareHistoryModal,
|
||||||
onToggleMediaVisibility,
|
onToggleMediaVisibility,
|
||||||
showMedia,
|
showMedia,
|
||||||
|
@ -93,23 +92,29 @@ const DetailedStatus: React.FC<IDetailedStatus> = ({
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
{(isUnderReview || isSensitive) ? (
|
{(isUnderReview || isSensitive) && (
|
||||||
<SensitiveContentOverlay
|
<SensitiveContentOverlay
|
||||||
status={status}
|
status={status}
|
||||||
visible={showMedia}
|
visible={showMedia}
|
||||||
onToggleVisibility={onToggleMediaVisibility}
|
onToggleVisibility={onToggleMediaVisibility}
|
||||||
/>
|
/>
|
||||||
) : null}
|
)}
|
||||||
|
|
||||||
<StatusContent status={actualStatus} />
|
<Stack space={4}>
|
||||||
|
<StatusContent status={actualStatus} />
|
||||||
|
|
||||||
<StatusMedia
|
{(quote || actualStatus.media_attachments.size > 0) && (
|
||||||
status={actualStatus}
|
<Stack space={4}>
|
||||||
showMedia={showMedia}
|
<StatusMedia
|
||||||
onToggleVisibility={onToggleMediaVisibility}
|
status={actualStatus}
|
||||||
/>
|
showMedia={showMedia}
|
||||||
|
onToggleVisibility={onToggleMediaVisibility}
|
||||||
|
/>
|
||||||
|
|
||||||
{quote}
|
{quote}
|
||||||
|
</Stack>
|
||||||
|
)}
|
||||||
|
</Stack>
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|
||||||
<HStack justifyContent='between' alignItems='center' className='py-2' wrap>
|
<HStack justifyContent='between' alignItems='center' className='py-2' wrap>
|
||||||
|
|
Binary file not shown.
|
@ -1,11 +1,11 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useIntl, MessageDescriptor } from 'react-intl';
|
import { useIntl, MessageDescriptor } from 'react-intl';
|
||||||
|
import { NotificationStack, NotificationObject, StyleFactoryFn } from 'react-notification';
|
||||||
import { useHistory } from 'react-router-dom';
|
import { useHistory } from 'react-router-dom';
|
||||||
|
|
||||||
import { dismissAlert } from 'soapbox/actions/alerts';
|
import { dismissAlert } from 'soapbox/actions/alerts';
|
||||||
import { Button } from 'soapbox/components/ui';
|
import { Button } from 'soapbox/components/ui';
|
||||||
import { useAppSelector, useAppDispatch } from 'soapbox/hooks';
|
import { useAppSelector, useAppDispatch } from 'soapbox/hooks';
|
||||||
import { NotificationStack, NotificationObject, StyleFactoryFn } from 'soapbox/react-notification';
|
|
||||||
|
|
||||||
import type { Alert } from 'soapbox/reducers/alerts';
|
import type { Alert } from 'soapbox/reducers/alerts';
|
||||||
|
|
||||||
|
|
|
@ -112,6 +112,7 @@ import {
|
||||||
TestTimeline,
|
TestTimeline,
|
||||||
LogoutPage,
|
LogoutPage,
|
||||||
AuthTokenList,
|
AuthTokenList,
|
||||||
|
ServiceWorkerInfo,
|
||||||
} from './util/async-components';
|
} from './util/async-components';
|
||||||
import { WrappedRoute } from './util/react_router_helpers';
|
import { WrappedRoute } from './util/react_router_helpers';
|
||||||
|
|
||||||
|
@ -311,6 +312,7 @@ const SwitchingColumnsArea: React.FC = ({ children }) => {
|
||||||
<WrappedRoute path='/developers/apps/create' developerOnly page={DefaultPage} component={CreateApp} content={children} />
|
<WrappedRoute path='/developers/apps/create' developerOnly page={DefaultPage} component={CreateApp} content={children} />
|
||||||
<WrappedRoute path='/developers/settings_store' developerOnly page={DefaultPage} component={SettingsStore} content={children} />
|
<WrappedRoute path='/developers/settings_store' developerOnly page={DefaultPage} component={SettingsStore} content={children} />
|
||||||
<WrappedRoute path='/developers/timeline' developerOnly page={DefaultPage} component={TestTimeline} content={children} />
|
<WrappedRoute path='/developers/timeline' developerOnly page={DefaultPage} component={TestTimeline} content={children} />
|
||||||
|
<WrappedRoute path='/developers/sw' developerOnly page={DefaultPage} component={ServiceWorkerInfo} content={children} />
|
||||||
<WrappedRoute path='/developers' page={DefaultPage} component={Developers} content={children} />
|
<WrappedRoute path='/developers' page={DefaultPage} component={Developers} content={children} />
|
||||||
<WrappedRoute path='/error/network' developerOnly page={EmptyPage} component={() => new Promise((_resolve, reject) => reject())} content={children} />
|
<WrappedRoute path='/error/network' developerOnly page={EmptyPage} component={() => new Promise((_resolve, reject) => reject())} content={children} />
|
||||||
<WrappedRoute path='/error' developerOnly page={EmptyPage} component={IntentionalError} content={children} />
|
<WrappedRoute path='/error' developerOnly page={EmptyPage} component={IntentionalError} content={children} />
|
||||||
|
|
|
@ -470,6 +470,10 @@ export function TestTimeline() {
|
||||||
return import(/* webpackChunkName: "features/test_timeline" */'../../test_timeline');
|
return import(/* webpackChunkName: "features/test_timeline" */'../../test_timeline');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function ServiceWorkerInfo() {
|
||||||
|
return import(/* webpackChunkName: "features/developers" */'../../developers/service-worker-info');
|
||||||
|
}
|
||||||
|
|
||||||
export function DatePicker() {
|
export function DatePicker() {
|
||||||
return import(/* webpackChunkName: "date_picker" */'../../birthdays/date_picker');
|
return import(/* webpackChunkName: "date_picker" */'../../birthdays/date_picker');
|
||||||
}
|
}
|
||||||
|
|
BIN
app/soapbox/react-notification/defaultPropTypes.js
vendored
BIN
app/soapbox/react-notification/defaultPropTypes.js
vendored
Binary file not shown.
88
app/soapbox/react-notification/index.d.ts
vendored
88
app/soapbox/react-notification/index.d.ts
vendored
|
@ -1,88 +0,0 @@
|
||||||
declare module 'soapbox/react-notification' {
|
|
||||||
import { Component, ReactElement } from 'react';
|
|
||||||
|
|
||||||
interface StyleFactoryFn {
|
|
||||||
(index: number, style: object | void, notification: NotificationProps): object;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface OnClickNotificationProps {
|
|
||||||
/**
|
|
||||||
* Callback function to run when the action is clicked.
|
|
||||||
* @param notification Notification currently being clicked
|
|
||||||
* @param deactivate Function that can be called to set the notification to inactive.
|
|
||||||
* Used to activate notification exit animation on click.
|
|
||||||
*/
|
|
||||||
onClick?(notification: NotificationProps, deactivate: () => void): void;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface NotificationProps extends OnClickNotificationProps {
|
|
||||||
/** The name of the action, e.g., "close" or "undo". */
|
|
||||||
action?: string;
|
|
||||||
/** Custom action styles. */
|
|
||||||
actionStyle?: object;
|
|
||||||
/** Custom snackbar styles when the bar is active. */
|
|
||||||
activeBarStyle?: object;
|
|
||||||
/**
|
|
||||||
* Custom class to apply to the top-level component when active.
|
|
||||||
* @default 'notification-bar-active'
|
|
||||||
*/
|
|
||||||
activeClassName?: string;
|
|
||||||
/** Custom snackbar styles. */
|
|
||||||
barStyle?: object;
|
|
||||||
/** Custom class to apply to the top-level component. */
|
|
||||||
className?: string;
|
|
||||||
/**
|
|
||||||
* Timeout for onDismiss event.
|
|
||||||
* @default 2000
|
|
||||||
*/
|
|
||||||
dismissAfter?: boolean | number;
|
|
||||||
/**
|
|
||||||
* If true, the notification is visible.
|
|
||||||
* @default false
|
|
||||||
*/
|
|
||||||
isActive?: boolean;
|
|
||||||
/** The message or component for the notification. */
|
|
||||||
message: string | ReactElement<NotificationProps>;
|
|
||||||
/** Setting this prop to `false` will disable all inline styles. */
|
|
||||||
style?: boolean;
|
|
||||||
/** The title for the notification. */
|
|
||||||
title?: string | ReactElement<any>;
|
|
||||||
/** Custom title styles. */
|
|
||||||
titleStyle?: object;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Callback function to run when dismissAfter timer runs out
|
|
||||||
* @param notification Notification currently being dismissed.
|
|
||||||
*/
|
|
||||||
onDismiss?(notification: NotificationProps): void;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface NotificationStackProps extends OnClickNotificationProps {
|
|
||||||
/** Create the style of the actions. */
|
|
||||||
actionStyleFactory?: StyleFactoryFn;
|
|
||||||
/** Create the style of the active notification. */
|
|
||||||
activeBarStyleFactory?: StyleFactoryFn;
|
|
||||||
/** Create the style of the notification. */
|
|
||||||
barStyleFactory?: StyleFactoryFn;
|
|
||||||
/**
|
|
||||||
* If false, notification dismiss timers start immediately.
|
|
||||||
* @default true
|
|
||||||
*/
|
|
||||||
dismissInOrder?: boolean;
|
|
||||||
/** Array of notifications to render. */
|
|
||||||
notifications: NotificationObject[];
|
|
||||||
/**
|
|
||||||
* Callback function to run when dismissAfter timer runs out
|
|
||||||
* @param notification Notification currently being dismissed.
|
|
||||||
*/
|
|
||||||
onDismiss?(notification: NotificationObject): void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface NotificationObject extends NotificationProps {
|
|
||||||
key: number | string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export class Notification extends Component<NotificationProps, {}> {}
|
|
||||||
|
|
||||||
export class NotificationStack extends Component<NotificationStackProps, {}> {}
|
|
||||||
}
|
|
BIN
app/soapbox/react-notification/index.js
vendored
BIN
app/soapbox/react-notification/index.js
vendored
Binary file not shown.
BIN
app/soapbox/react-notification/notification.js
vendored
BIN
app/soapbox/react-notification/notification.js
vendored
Binary file not shown.
BIN
app/soapbox/react-notification/notificationStack.js
vendored
BIN
app/soapbox/react-notification/notificationStack.js
vendored
Binary file not shown.
Binary file not shown.
|
@ -1,16 +0,0 @@
|
||||||
import { Map as ImmutableMap } from 'immutable';
|
|
||||||
|
|
||||||
import reducer from '../group_editor';
|
|
||||||
|
|
||||||
describe('group_editor reducer', () => {
|
|
||||||
it('should return the initial state', () => {
|
|
||||||
expect(reducer(undefined, {} as any)).toEqual(ImmutableMap({
|
|
||||||
groupId: null,
|
|
||||||
isSubmitting: false,
|
|
||||||
isChanged: false,
|
|
||||||
title: '',
|
|
||||||
description: '',
|
|
||||||
coverImage: null,
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,13 +0,0 @@
|
||||||
import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
|
|
||||||
|
|
||||||
import reducer from '../group_lists';
|
|
||||||
|
|
||||||
describe('group_lists reducer', () => {
|
|
||||||
it('should return the initial state', () => {
|
|
||||||
expect(reducer(undefined, {} as any)).toEqual(ImmutableMap({
|
|
||||||
featured: ImmutableList(),
|
|
||||||
member: ImmutableList(),
|
|
||||||
admin: ImmutableList(),
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,9 +0,0 @@
|
||||||
import { Map as ImmutableMap } from 'immutable';
|
|
||||||
|
|
||||||
import reducer from '../group_relationships';
|
|
||||||
|
|
||||||
describe('group_relationships reducer', () => {
|
|
||||||
it('should return the initial state', () => {
|
|
||||||
expect(reducer(undefined, {} as any)).toEqual(ImmutableMap());
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -1,9 +0,0 @@
|
||||||
import { Map as ImmutableMap } from 'immutable';
|
|
||||||
|
|
||||||
import reducer from '../groups';
|
|
||||||
|
|
||||||
describe('groups reducer', () => {
|
|
||||||
it('should return the initial state', () => {
|
|
||||||
expect(reducer(undefined, {} as any)).toEqual(ImmutableMap());
|
|
||||||
});
|
|
||||||
});
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -14,8 +14,6 @@ describe('user_lists reducer', () => {
|
||||||
blocks: { next: null, items: ImmutableOrderedSet(), isLoading: false },
|
blocks: { next: null, items: ImmutableOrderedSet(), isLoading: false },
|
||||||
mutes: { next: null, items: ImmutableOrderedSet(), isLoading: false },
|
mutes: { next: null, items: ImmutableOrderedSet(), isLoading: false },
|
||||||
directory: { next: null, items: ImmutableOrderedSet(), isLoading: true },
|
directory: { next: null, items: ImmutableOrderedSet(), isLoading: true },
|
||||||
groups: {},
|
|
||||||
groups_removed_accounts: {},
|
|
||||||
pinned: {},
|
pinned: {},
|
||||||
birthday_reminders: {},
|
birthday_reminders: {},
|
||||||
familiar_followers: {},
|
familiar_followers: {},
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -25,10 +25,6 @@ import custom_emojis from './custom_emojis';
|
||||||
import domain_lists from './domain_lists';
|
import domain_lists from './domain_lists';
|
||||||
import dropdown_menu from './dropdown_menu';
|
import dropdown_menu from './dropdown_menu';
|
||||||
import filters from './filters';
|
import filters from './filters';
|
||||||
import group_editor from './group_editor';
|
|
||||||
import group_lists from './group_lists';
|
|
||||||
import group_relationships from './group_relationships';
|
|
||||||
import groups from './groups';
|
|
||||||
import history from './history';
|
import history from './history';
|
||||||
import instance from './instance';
|
import instance from './instance';
|
||||||
import listAdder from './list_adder';
|
import listAdder from './list_adder';
|
||||||
|
@ -95,10 +91,6 @@ const reducers = {
|
||||||
suggestions,
|
suggestions,
|
||||||
polls,
|
polls,
|
||||||
trends,
|
trends,
|
||||||
groups,
|
|
||||||
group_relationships,
|
|
||||||
group_lists,
|
|
||||||
group_editor,
|
|
||||||
sidebar,
|
sidebar,
|
||||||
patron,
|
patron,
|
||||||
soapbox,
|
soapbox,
|
||||||
|
|
|
@ -12,7 +12,6 @@ import {
|
||||||
ACCOUNT_MUTE_SUCCESS,
|
ACCOUNT_MUTE_SUCCESS,
|
||||||
ACCOUNT_UNFOLLOW_SUCCESS,
|
ACCOUNT_UNFOLLOW_SUCCESS,
|
||||||
} from '../actions/accounts';
|
} from '../actions/accounts';
|
||||||
import { GROUP_REMOVE_STATUS_SUCCESS } from '../actions/groups';
|
|
||||||
import {
|
import {
|
||||||
STATUS_CREATE_REQUEST,
|
STATUS_CREATE_REQUEST,
|
||||||
STATUS_CREATE_SUCCESS,
|
STATUS_CREATE_SUCCESS,
|
||||||
|
@ -210,10 +209,6 @@ const filterTimelines = (state: State, relationship: APIEntity, statuses: Immuta
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const removeStatusFromGroup = (state: State, groupId: string, statusId: string) => {
|
|
||||||
return state.updateIn([`group:${groupId}`, 'items'], ImmutableOrderedSet(), ids => (ids as ImmutableOrderedSet<string>).delete(statusId));
|
|
||||||
};
|
|
||||||
|
|
||||||
const timelineDequeue = (state: State, timelineId: string) => {
|
const timelineDequeue = (state: State, timelineId: string) => {
|
||||||
const top = state.getIn([timelineId, 'top']);
|
const top = state.getIn([timelineId, 'top']);
|
||||||
|
|
||||||
|
@ -348,8 +343,6 @@ export default function timelines(state: State = initialState, action: AnyAction
|
||||||
return timelineConnect(state, action.timeline);
|
return timelineConnect(state, action.timeline);
|
||||||
case TIMELINE_DISCONNECT:
|
case TIMELINE_DISCONNECT:
|
||||||
return timelineDisconnect(state, action.timeline);
|
return timelineDisconnect(state, action.timeline);
|
||||||
case GROUP_REMOVE_STATUS_SUCCESS:
|
|
||||||
return removeStatusFromGroup(state, action.groupId, action.id);
|
|
||||||
case TIMELINE_REPLACE:
|
case TIMELINE_REPLACE:
|
||||||
return state
|
return state
|
||||||
.update('home', TimelineRecord(), timeline => timeline.withMutations(timeline => {
|
.update('home', TimelineRecord(), timeline => timeline.withMutations(timeline => {
|
||||||
|
|
|
@ -32,13 +32,6 @@ import {
|
||||||
import {
|
import {
|
||||||
FAMILIAR_FOLLOWERS_FETCH_SUCCESS,
|
FAMILIAR_FOLLOWERS_FETCH_SUCCESS,
|
||||||
} from '../actions/familiar_followers';
|
} from '../actions/familiar_followers';
|
||||||
import {
|
|
||||||
GROUP_MEMBERS_FETCH_SUCCESS,
|
|
||||||
GROUP_MEMBERS_EXPAND_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_FETCH_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_EXPAND_SUCCESS,
|
|
||||||
GROUP_REMOVED_ACCOUNTS_REMOVE_SUCCESS,
|
|
||||||
} from '../actions/groups';
|
|
||||||
import {
|
import {
|
||||||
REBLOGS_FETCH_SUCCESS,
|
REBLOGS_FETCH_SUCCESS,
|
||||||
FAVOURITES_FETCH_SUCCESS,
|
FAVOURITES_FETCH_SUCCESS,
|
||||||
|
@ -82,8 +75,6 @@ export const ReducerRecord = ImmutableRecord({
|
||||||
blocks: ListRecord(),
|
blocks: ListRecord(),
|
||||||
mutes: ListRecord(),
|
mutes: ListRecord(),
|
||||||
directory: ListRecord({ isLoading: true }),
|
directory: ListRecord({ isLoading: true }),
|
||||||
groups: ImmutableMap<string, List>(),
|
|
||||||
groups_removed_accounts: ImmutableMap<string, List>(),
|
|
||||||
pinned: ImmutableMap<string, List>(),
|
pinned: ImmutableMap<string, List>(),
|
||||||
birthday_reminders: ImmutableMap<string, List>(),
|
birthday_reminders: ImmutableMap<string, List>(),
|
||||||
familiar_followers: ImmutableMap<string, List>(),
|
familiar_followers: ImmutableMap<string, List>(),
|
||||||
|
@ -94,7 +85,7 @@ export type List = ReturnType<typeof ListRecord>;
|
||||||
type Reaction = ReturnType<typeof ReactionRecord>;
|
type Reaction = ReturnType<typeof ReactionRecord>;
|
||||||
type ReactionList = ReturnType<typeof ReactionListRecord>;
|
type ReactionList = ReturnType<typeof ReactionListRecord>;
|
||||||
type Items = ImmutableOrderedSet<string>;
|
type Items = ImmutableOrderedSet<string>;
|
||||||
type NestedListPath = ['followers' | 'following' | 'reblogged_by' | 'favourited_by' | 'reactions' | 'groups' | 'groups_removed_accounts' | 'pinned' | 'birthday_reminders' | 'familiar_followers', string];
|
type NestedListPath = ['followers' | 'following' | 'reblogged_by' | 'favourited_by' | 'reactions' | 'pinned' | 'birthday_reminders' | 'familiar_followers', string];
|
||||||
type ListPath = ['follow_requests' | 'blocks' | 'mutes' | 'directory'];
|
type ListPath = ['follow_requests' | 'blocks' | 'mutes' | 'directory'];
|
||||||
|
|
||||||
const normalizeList = (state: State, path: NestedListPath | ListPath, accounts: APIEntity[], next?: string | null) => {
|
const normalizeList = (state: State, path: NestedListPath | ListPath, accounts: APIEntity[], next?: string | null) => {
|
||||||
|
@ -170,16 +161,6 @@ export default function userLists(state = ReducerRecord(), action: AnyAction) {
|
||||||
case DIRECTORY_FETCH_FAIL:
|
case DIRECTORY_FETCH_FAIL:
|
||||||
case DIRECTORY_EXPAND_FAIL:
|
case DIRECTORY_EXPAND_FAIL:
|
||||||
return state.setIn(['directory', 'isLoading'], false);
|
return state.setIn(['directory', 'isLoading'], false);
|
||||||
case GROUP_MEMBERS_FETCH_SUCCESS:
|
|
||||||
return normalizeList(state, ['groups', action.id], action.accounts, action.next);
|
|
||||||
case GROUP_MEMBERS_EXPAND_SUCCESS:
|
|
||||||
return appendToList(state, ['groups', action.id], action.accounts, action.next);
|
|
||||||
case GROUP_REMOVED_ACCOUNTS_FETCH_SUCCESS:
|
|
||||||
return normalizeList(state, ['groups_removed_accounts', action.id], action.accounts, action.next);
|
|
||||||
case GROUP_REMOVED_ACCOUNTS_EXPAND_SUCCESS:
|
|
||||||
return appendToList(state, ['groups_removed_accounts', action.id], action.accounts, action.next);
|
|
||||||
case GROUP_REMOVED_ACCOUNTS_REMOVE_SUCCESS:
|
|
||||||
return removeFromList(state, ['groups_removed_accounts', action.groupId], action.id);
|
|
||||||
case PINNED_ACCOUNTS_FETCH_SUCCESS:
|
case PINNED_ACCOUNTS_FETCH_SUCCESS:
|
||||||
return normalizeList(state, ['pinned', action.id], action.accounts, action.next);
|
return normalizeList(state, ['pinned', action.id], action.accounts, action.next);
|
||||||
case BIRTHDAY_REMINDERS_FETCH_SUCCESS:
|
case BIRTHDAY_REMINDERS_FETCH_SUCCESS:
|
||||||
|
|
15
app/soapbox/utils/sw.ts
Normal file
15
app/soapbox/utils/sw.ts
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
/** Unregister the ServiceWorker */
|
||||||
|
// https://stackoverflow.com/a/49771828/8811886
|
||||||
|
const unregisterSw = async(): Promise<void> => {
|
||||||
|
if (navigator.serviceWorker) {
|
||||||
|
// FIXME: this only works if using a single tab.
|
||||||
|
// Send a message to sw.js instead to refresh all tabs.
|
||||||
|
const registrations = await navigator.serviceWorker.getRegistrations();
|
||||||
|
const unregisterAll = registrations.map(r => r.unregister());
|
||||||
|
await Promise.all(unregisterAll);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export {
|
||||||
|
unregisterSw,
|
||||||
|
};
|
|
@ -1,6 +1,5 @@
|
||||||
.media-gallery {
|
.media-gallery {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
margin-top: 8px;
|
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
|
|
@ -10,7 +10,6 @@
|
||||||
.detailed-status {
|
.detailed-status {
|
||||||
.reply-mentions {
|
.reply-mentions {
|
||||||
display: block;
|
display: block;
|
||||||
margin: 4px 0 0 0;
|
|
||||||
|
|
||||||
span {
|
span {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
|
|
@ -14,11 +14,6 @@
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
animation: fade 150ms linear;
|
animation: fade 150ms linear;
|
||||||
|
|
||||||
.video-player,
|
|
||||||
.audio-player {
|
|
||||||
margin-top: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.light {
|
&.light {
|
||||||
.display-name {
|
.display-name {
|
||||||
strong {
|
strong {
|
||||||
|
|
|
@ -7,7 +7,6 @@
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
margin-top: 8px;
|
|
||||||
position: relative;
|
position: relative;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
|
|
|
@ -66,7 +66,7 @@
|
||||||
"@sentry/browser": "^7.11.1",
|
"@sentry/browser": "^7.11.1",
|
||||||
"@sentry/react": "^7.11.1",
|
"@sentry/react": "^7.11.1",
|
||||||
"@sentry/tracing": "^7.11.1",
|
"@sentry/tracing": "^7.11.1",
|
||||||
"@tabler/icons": "^1.73.0",
|
"@tabler/icons": "^1.109.0",
|
||||||
"@tailwindcss/forms": "^0.5.3",
|
"@tailwindcss/forms": "^0.5.3",
|
||||||
"@tailwindcss/typography": "^0.5.7",
|
"@tailwindcss/typography": "^0.5.7",
|
||||||
"@tanstack/react-query": "^4.0.10",
|
"@tanstack/react-query": "^4.0.10",
|
||||||
|
@ -166,6 +166,7 @@
|
||||||
"react-inlinesvg": "^3.0.0",
|
"react-inlinesvg": "^3.0.0",
|
||||||
"react-intl": "^5.0.0",
|
"react-intl": "^5.0.0",
|
||||||
"react-motion": "^0.5.2",
|
"react-motion": "^0.5.2",
|
||||||
|
"react-notification": "^6.8.5",
|
||||||
"react-otp-input": "^2.4.0",
|
"react-otp-input": "^2.4.0",
|
||||||
"react-overlays": "^0.9.0",
|
"react-overlays": "^0.9.0",
|
||||||
"react-popper": "^2.3.0",
|
"react-popper": "^2.3.0",
|
||||||
|
|
15
yarn.lock
15
yarn.lock
|
@ -2271,10 +2271,10 @@
|
||||||
remark "^13.0.0"
|
remark "^13.0.0"
|
||||||
unist-util-find-all-after "^3.0.2"
|
unist-util-find-all-after "^3.0.2"
|
||||||
|
|
||||||
"@tabler/icons@^1.73.0":
|
"@tabler/icons@^1.109.0":
|
||||||
version "1.73.0"
|
version "1.109.0"
|
||||||
resolved "https://registry.yarnpkg.com/@tabler/icons/-/icons-1.73.0.tgz#26d81858baf41be939504e1f9b4b32835eda6fdb"
|
resolved "https://registry.yarnpkg.com/@tabler/icons/-/icons-1.109.0.tgz#11626c3fc097f2f70c4c197e4b9909fb05380752"
|
||||||
integrity sha512-MhAHFzVj79ZWlAIRD++7Mk55PZsdlEdkfkjO3DD257mqj8iJZQRAQtkx2UFJXVs2mMrcOUu1qtj4rlVC8BfnKA==
|
integrity sha512-B0YetE4pB6HY2Wa57v/LJ3NgkJzKYPze4U0DurIqPoKSptatKv2ga76FZSkO6EUpkYfHMtGPM6QjpJljfuCmAQ==
|
||||||
|
|
||||||
"@tailwindcss/forms@^0.5.3":
|
"@tailwindcss/forms@^0.5.3":
|
||||||
version "0.5.3"
|
version "0.5.3"
|
||||||
|
@ -9922,6 +9922,13 @@ react-motion@^0.5.2:
|
||||||
prop-types "^15.5.8"
|
prop-types "^15.5.8"
|
||||||
raf "^3.1.0"
|
raf "^3.1.0"
|
||||||
|
|
||||||
|
react-notification@^6.8.5:
|
||||||
|
version "6.8.5"
|
||||||
|
resolved "https://registry.yarnpkg.com/react-notification/-/react-notification-6.8.5.tgz#7ea90a633bb2a280d899e30c93cf372265cce4f0"
|
||||||
|
integrity sha512-3pJPhSsWNYizpyeMeWuC+jVthqE9WKqQ6rHq2naiiP4fLGN4irwL2Xp2Q8Qn7agW/e4BIDxarab6fJOUp1cKUw==
|
||||||
|
dependencies:
|
||||||
|
prop-types "^15.6.2"
|
||||||
|
|
||||||
react-onclickoutside@^6.12.0:
|
react-onclickoutside@^6.12.0:
|
||||||
version "6.12.1"
|
version "6.12.1"
|
||||||
resolved "https://registry.yarnpkg.com/react-onclickoutside/-/react-onclickoutside-6.12.1.tgz#92dddd28f55e483a1838c5c2930e051168c1e96b"
|
resolved "https://registry.yarnpkg.com/react-onclickoutside/-/react-onclickoutside-6.12.1.tgz#92dddd28f55e483a1838c5c2930e051168c1e96b"
|
||||||
|
|
Loading…
Reference in a new issue