pl-fe: remove some immutable usage
Signed-off-by: marcin mikołajczak <git@mkljczk.pl>
This commit is contained in:
parent
66fbdf2ad9
commit
865130c55a
15 changed files with 197 additions and 169 deletions
|
@ -59,7 +59,7 @@ const Aliases = () => {
|
|||
</CardHeader>
|
||||
<Search />
|
||||
{
|
||||
loaded && searchAccountIds.size === 0 ? (
|
||||
loaded && searchAccountIds.length === 0 ? (
|
||||
<div className='empty-column-indicator'>
|
||||
<FormattedMessage id='empty_column.aliases.suggestions' defaultMessage='There are no account suggestions available for the provided term.' />
|
||||
</div>
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
import { Map as ImmutableMap } from 'immutable';
|
||||
import React, { useState, useEffect } from 'react';
|
||||
import { useIntl, defineMessages, FormattedMessage } from 'react-intl';
|
||||
|
||||
|
@ -19,7 +18,7 @@ interface ICaptchaField {
|
|||
name?: string;
|
||||
value: string;
|
||||
onChange?: React.ChangeEventHandler<HTMLInputElement>;
|
||||
onFetch?: (captcha: ImmutableMap<string, any>) => void;
|
||||
onFetch?: (captcha: Record<string, any>) => void;
|
||||
onFetchFail?: (error: Error) => void;
|
||||
onClick?: React.MouseEventHandler;
|
||||
refreshInterval?: number;
|
||||
|
@ -38,12 +37,11 @@ const CaptchaField: React.FC<ICaptchaField> = ({
|
|||
}) => {
|
||||
const dispatch = useAppDispatch();
|
||||
|
||||
const [captcha, setCaptcha] = useState(ImmutableMap<string, any>());
|
||||
const [captcha, setCaptcha] = useState<Record<string, any>>({});
|
||||
const [refresh, setRefresh] = useState<NodeJS.Timeout | undefined>(undefined);
|
||||
|
||||
const getCaptcha = () => {
|
||||
dispatch(fetchCaptcha()).then((response) => {
|
||||
const captcha = ImmutableMap(response);
|
||||
dispatch(fetchCaptcha()).then((captcha) => {
|
||||
setCaptcha(captcha);
|
||||
onFetch(captcha);
|
||||
}).catch((error: Error) => {
|
||||
|
@ -98,7 +96,7 @@ const CaptchaField: React.FC<ICaptchaField> = ({
|
|||
};
|
||||
|
||||
interface INativeCaptchaField {
|
||||
captcha: ImmutableMap<string, any>;
|
||||
captcha: Record<string, any>;
|
||||
onChange: React.ChangeEventHandler<HTMLInputElement>;
|
||||
onClick: React.MouseEventHandler;
|
||||
name?: string;
|
||||
|
@ -111,7 +109,7 @@ const NativeCaptchaField: React.FC<INativeCaptchaField> = ({ captcha, onChange,
|
|||
return (
|
||||
<Stack space={2}>
|
||||
<div className='flex w-full items-center justify-center rounded-md border border-solid border-gray-300 bg-white dark:border-gray-600'>
|
||||
<img alt={intl.formatMessage(messages.captcha)} src={captcha.get('url')} onClick={onClick} />
|
||||
<img alt={intl.formatMessage(messages.captcha)} src={captcha.url} onClick={onClick} />
|
||||
</div>
|
||||
|
||||
<Input
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
import { Map as ImmutableMap } from 'immutable';
|
||||
import debounce from 'lodash/debounce';
|
||||
import React, { useState, useRef, useCallback } from 'react';
|
||||
import { useIntl, FormattedMessage, defineMessages } from 'react-intl';
|
||||
|
@ -221,12 +220,12 @@ const RegistrationForm: React.FC<IRegistrationForm> = ({ inviteToken }) => {
|
|||
refreshCaptcha();
|
||||
};
|
||||
|
||||
const onFetchCaptcha = (captcha: ImmutableMap<string, any>) => {
|
||||
const onFetchCaptcha = (captcha: Record<string, any>) => {
|
||||
setCaptchaLoading(false);
|
||||
setParams(params => ({
|
||||
...params,
|
||||
captcha_token: captcha.get('token'),
|
||||
captcha_answer_data: captcha.get('answer_data'),
|
||||
captcha_token: captcha.token,
|
||||
captcha_answer_data: captcha.answer_data,
|
||||
}));
|
||||
};
|
||||
|
||||
|
|
|
@ -2,12 +2,10 @@
|
|||
// See: https://github.com/spothq/cryptocurrency-icons/blob/master/manifest.json
|
||||
|
||||
import manifest from 'cryptocurrency-icons/manifest.json';
|
||||
import { List as ImmutableList, Map as ImmutableMap, fromJS } from 'immutable';
|
||||
|
||||
const manifestMap = (fromJS(manifest) as ImmutableList<ImmutableMap<string, string>>).reduce(
|
||||
(acc: ImmutableMap<string, ImmutableMap<string, string>>, entry: ImmutableMap<string, string>) =>
|
||||
acc.set(entry.get('symbol')!.toLowerCase(), entry),
|
||||
ImmutableMap(),
|
||||
).toJS();
|
||||
const manifestMap = manifest.reduce((acc: Record<string, typeof manifest[0]>, entry) => {
|
||||
acc[entry.symbol.toLowerCase()] = entry;
|
||||
return acc;
|
||||
}, {});
|
||||
|
||||
export { manifestMap as default };
|
||||
|
|
|
@ -11,8 +11,6 @@ import { federationRestrictionsDisclosed } from 'pl-fe/utils/state';
|
|||
|
||||
import RestrictedInstance from './components/restricted-instance';
|
||||
|
||||
import type { OrderedSet as ImmutableOrderedSet } from 'immutable';
|
||||
|
||||
const messages = defineMessages({
|
||||
heading: { id: 'column.federation_restrictions', defaultMessage: 'Federation Restrictions' },
|
||||
boxTitle: { id: 'federation_restrictions.explanation_box.title', defaultMessage: 'Instance-specific policies' },
|
||||
|
@ -27,7 +25,7 @@ const FederationRestrictions = () => {
|
|||
|
||||
const getHosts = useCallback(makeGetHosts(), []);
|
||||
|
||||
const hosts = useAppSelector((state) => getHosts(state)) as ImmutableOrderedSet<string>;
|
||||
const hosts = useAppSelector((state) => getHosts(state));
|
||||
const disclosed = useAppSelector((state) => federationRestrictionsDisclosed(state));
|
||||
|
||||
const [explanationBoxExpanded, setExplanationBoxExpanded] = useState(true);
|
||||
|
|
|
@ -1,5 +1,4 @@
|
|||
import clsx from 'clsx';
|
||||
import { List as ImmutableList } from 'immutable';
|
||||
import debounce from 'lodash/debounce';
|
||||
import React, { useCallback, useEffect, useRef } from 'react';
|
||||
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
||||
|
@ -31,8 +30,8 @@ const messages = defineMessages({
|
|||
});
|
||||
|
||||
const getNotifications = createSelector([
|
||||
(state: RootState) => state.notifications.items.toList(),
|
||||
], (notifications) => notifications.filter(item => item !== null));
|
||||
(state: RootState) => state.notifications.items.toArray(),
|
||||
], (notifications) => notifications.map(([_, notification]) => notification).filter(item => item !== null));
|
||||
|
||||
const Notifications = () => {
|
||||
const dispatch = useAppDispatch();
|
||||
|
@ -47,7 +46,7 @@ const Notifications = () => {
|
|||
const hasMore = useAppSelector(state => state.notifications.hasMore);
|
||||
const totalQueuedNotificationsCount = useAppSelector(state => state.notifications.totalQueuedNotificationsCount || 0);
|
||||
|
||||
const scrollableContentRef = useRef<ImmutableList<JSX.Element> | null>(null);
|
||||
const scrollableContentRef = useRef<Array<JSX.Element> | null>(null);
|
||||
|
||||
// const handleLoadGap = (maxId) => {
|
||||
// dispatch(expandNotifications({ maxId }));
|
||||
|
@ -105,7 +104,7 @@ const Notifications = () => {
|
|||
? <FormattedMessage id='empty_column.notifications' defaultMessage="You don't have any notifications yet. Interact with others to start the conversation." />
|
||||
: <FormattedMessage id='empty_column.notifications_filtered' defaultMessage="You don't have any notifications of this type yet." />;
|
||||
|
||||
let scrollableContent: ImmutableList<JSX.Element> | null = null;
|
||||
let scrollableContent: Array<JSX.Element> | null = null;
|
||||
|
||||
const filterBarContainer = showFilterBar
|
||||
? (<FilterBar />)
|
||||
|
@ -113,7 +112,7 @@ const Notifications = () => {
|
|||
|
||||
if (isLoading && scrollableContentRef.current) {
|
||||
scrollableContent = scrollableContentRef.current;
|
||||
} else if (notifications.size > 0 || hasMore) {
|
||||
} else if (notifications.length > 0 || hasMore) {
|
||||
scrollableContent = notifications.map((item) => (
|
||||
<Notification
|
||||
key={item.group_key}
|
||||
|
@ -131,7 +130,7 @@ const Notifications = () => {
|
|||
const scrollContainer = (
|
||||
<ScrollableList
|
||||
isLoading={isLoading}
|
||||
showLoading={isLoading && notifications.size === 0}
|
||||
showLoading={isLoading && notifications.length === 0}
|
||||
hasMore={hasMore}
|
||||
emptyMessage={emptyMessage}
|
||||
placeholderComponent={PlaceholderNotification}
|
||||
|
@ -139,7 +138,7 @@ const Notifications = () => {
|
|||
onLoadMore={handleLoadOlder}
|
||||
onScroll={handleScroll}
|
||||
listClassName={clsx('divide-y divide-solid divide-gray-200 black:divide-gray-800 dark:divide-primary-800', {
|
||||
'animate-pulse': notifications.size === 0,
|
||||
'animate-pulse': notifications.length === 0,
|
||||
})}
|
||||
>
|
||||
{scrollableContent!}
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
import { Record as ImmutableRecord } from 'immutable';
|
||||
import { MediaAttachment } from 'pl-api';
|
||||
import React, { useState } from 'react';
|
||||
|
||||
|
@ -7,12 +6,12 @@ interface IPlaceholderMediaGallery {
|
|||
defaultWidth?: number;
|
||||
}
|
||||
|
||||
const SizeData = ImmutableRecord({
|
||||
style: {} as React.CSSProperties,
|
||||
itemsDimensions: [] as Record<string, string>[],
|
||||
size: 1 as number,
|
||||
width: 0 as number,
|
||||
});
|
||||
interface SizeData {
|
||||
style: React.CSSProperties;
|
||||
itemsDimensions: Record<string, string>[];
|
||||
size: number;
|
||||
width: number;
|
||||
}
|
||||
|
||||
const PlaceholderMediaGallery: React.FC<IPlaceholderMediaGallery> = ({ media, defaultWidth }) => {
|
||||
const [width, setWidth] = useState(defaultWidth);
|
||||
|
@ -23,7 +22,7 @@ const PlaceholderMediaGallery: React.FC<IPlaceholderMediaGallery> = ({ media, de
|
|||
}
|
||||
};
|
||||
|
||||
const getSizeData = (size: number) => {
|
||||
const getSizeData = (size: number): SizeData => {
|
||||
const style: React.CSSProperties = {};
|
||||
let itemsDimensions: Record<string, string>[] = [];
|
||||
|
||||
|
@ -59,12 +58,12 @@ const PlaceholderMediaGallery: React.FC<IPlaceholderMediaGallery> = ({ media, de
|
|||
];
|
||||
}
|
||||
|
||||
return SizeData({
|
||||
return {
|
||||
style,
|
||||
itemsDimensions,
|
||||
size,
|
||||
width,
|
||||
});
|
||||
width: width || 0,
|
||||
};
|
||||
};
|
||||
|
||||
const renderItem = (dimensions: Record<string, string>, i: number) => {
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
import { Set as ImmutableSet } from 'immutable';
|
||||
import React from 'react';
|
||||
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
||||
|
||||
|
@ -106,7 +105,7 @@ const Preferences = () => {
|
|||
};
|
||||
|
||||
const onSelectMultiple = (selectedList: string[], path: string[]) => {
|
||||
dispatch(changeSetting(path, ImmutableSet(selectedList.sort((a, b) => a.localeCompare(b))), { showAlert: true }));
|
||||
dispatch(changeSetting(path, selectedList.toSorted((a, b) => a.localeCompare(b)), { showAlert: true }));
|
||||
};
|
||||
|
||||
const onToggleChange = (key: string[], checked: boolean) => {
|
||||
|
|
|
@ -54,7 +54,7 @@ const ListEditorModal: React.FC<BaseModalProps & ListEditorModalProps> = ({ list
|
|||
<EditListForm />
|
||||
<br />
|
||||
|
||||
{accountIds.size > 0 && (
|
||||
{accountIds.length > 0 && (
|
||||
<div>
|
||||
<CardHeader>
|
||||
<CardTitle title={intl.formatMessage(messages.removeFromList)} />
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { List as ImmutableList, Record as ImmutableRecord } from 'immutable';
|
||||
import { create } from 'mutative';
|
||||
|
||||
import {
|
||||
ALIASES_SUGGESTIONS_READY,
|
||||
|
@ -8,37 +8,52 @@ import {
|
|||
AliasesAction,
|
||||
} from '../actions/aliases';
|
||||
|
||||
const ReducerRecord = ImmutableRecord({
|
||||
aliases: ImmutableRecord({
|
||||
items: ImmutableList<string>(),
|
||||
interface State {
|
||||
aliases: {
|
||||
items: Array<string>;
|
||||
loaded: boolean;
|
||||
};
|
||||
suggestions: {
|
||||
items: Array<string>;
|
||||
value: string;
|
||||
loaded: boolean;
|
||||
};
|
||||
}
|
||||
|
||||
const initialState: State = {
|
||||
aliases: {
|
||||
items: [],
|
||||
loaded: false,
|
||||
})(),
|
||||
suggestions: ImmutableRecord({
|
||||
items: ImmutableList<string>(),
|
||||
},
|
||||
suggestions: {
|
||||
items: [],
|
||||
value: '',
|
||||
loaded: false,
|
||||
})(),
|
||||
});
|
||||
},
|
||||
};
|
||||
|
||||
const aliasesReducer = (state = ReducerRecord(), action: AliasesAction) => {
|
||||
const aliasesReducer = (state = initialState, action: AliasesAction): State => {
|
||||
switch (action.type) {
|
||||
case ALIASES_FETCH_SUCCESS:
|
||||
return state
|
||||
.setIn(['aliases', 'items'], action.value);
|
||||
return create(state, (draft) => {
|
||||
draft.aliases.items = action.value;
|
||||
});
|
||||
case ALIASES_SUGGESTIONS_CHANGE:
|
||||
return state
|
||||
.setIn(['suggestions', 'value'], action.value)
|
||||
.setIn(['suggestions', 'loaded'], false);
|
||||
return create(state, (draft) => {
|
||||
draft.suggestions.value = action.value;
|
||||
draft.suggestions.loaded = false;
|
||||
});
|
||||
case ALIASES_SUGGESTIONS_READY:
|
||||
return state
|
||||
.setIn(['suggestions', 'items'], ImmutableList(action.accounts.map((item) => item.id)))
|
||||
.setIn(['suggestions', 'loaded'], true);
|
||||
return create(state, (draft) => {
|
||||
draft.suggestions.items = action.accounts.map((item) => item.id);
|
||||
draft.suggestions.loaded = true;
|
||||
});
|
||||
case ALIASES_SUGGESTIONS_CLEAR:
|
||||
return state.update('suggestions', suggestions => suggestions.withMutations(map => {
|
||||
map.set('items', ImmutableList());
|
||||
map.set('value', '');
|
||||
map.set('loaded', false);
|
||||
}));
|
||||
return create(state, (draft) => {
|
||||
draft.suggestions.items = [];
|
||||
draft.suggestions.value = '';
|
||||
draft.suggestions.loaded = false;
|
||||
});
|
||||
default:
|
||||
return state;
|
||||
}
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { List as ImmutableList, Record as ImmutableRecord } from 'immutable';
|
||||
import { create } from 'mutative';
|
||||
|
||||
import {
|
||||
LIST_ADDER_RESET,
|
||||
|
@ -12,42 +12,54 @@ import {
|
|||
|
||||
import type { AnyAction } from 'redux';
|
||||
|
||||
const ListsRecord = ImmutableRecord({
|
||||
items: ImmutableList<string>(),
|
||||
interface State {
|
||||
accountId: string | null;
|
||||
lists: {
|
||||
items: Array<string>;
|
||||
loaded: boolean;
|
||||
isLoading: boolean;
|
||||
};
|
||||
}
|
||||
|
||||
const initialState: State = {
|
||||
accountId: null,
|
||||
lists: {
|
||||
items: [],
|
||||
loaded: false,
|
||||
isLoading: false,
|
||||
});
|
||||
},
|
||||
};
|
||||
|
||||
const ReducerRecord = ImmutableRecord({
|
||||
accountId: null as string | null,
|
||||
|
||||
lists: ListsRecord(),
|
||||
});
|
||||
|
||||
type State = ReturnType<typeof ReducerRecord>;
|
||||
|
||||
const listAdderReducer = (state: State = ReducerRecord(), action: AnyAction) => {
|
||||
const listAdderReducer = (state: State = initialState, action: AnyAction): State => {
|
||||
switch (action.type) {
|
||||
case LIST_ADDER_RESET:
|
||||
return ReducerRecord();
|
||||
return initialState;
|
||||
case LIST_ADDER_SETUP:
|
||||
return state.withMutations(map => {
|
||||
map.set('accountId', action.account.id);
|
||||
return create(state, (draft) => {
|
||||
draft.accountId = action.account.id;
|
||||
});
|
||||
case LIST_ADDER_LISTS_FETCH_REQUEST:
|
||||
return state.setIn(['lists', 'isLoading'], true);
|
||||
return create(state, (draft) => {
|
||||
draft.lists.isLoading = true;
|
||||
});
|
||||
case LIST_ADDER_LISTS_FETCH_FAIL:
|
||||
return state.setIn(['lists', 'isLoading'], false);
|
||||
return create(state, (draft) => {
|
||||
draft.lists.isLoading = false;
|
||||
});
|
||||
case LIST_ADDER_LISTS_FETCH_SUCCESS:
|
||||
return state.update('lists', lists => lists.withMutations(map => {
|
||||
map.set('isLoading', false);
|
||||
map.set('loaded', true);
|
||||
map.set('items', ImmutableList(action.lists.map((item: { id: string }) => item.id)));
|
||||
}));
|
||||
return create(state, (draft) => {
|
||||
draft.lists.isLoading = false;
|
||||
draft.lists.loaded = true;
|
||||
draft.lists.items = action.lists.map((item: { id: string }) => item.id);
|
||||
});
|
||||
case LIST_EDITOR_ADD_SUCCESS:
|
||||
return state.updateIn(['lists', 'items'], list => (list as ImmutableList<string>).unshift(action.listId));
|
||||
return create(state, (draft) => {
|
||||
draft.lists.items = [action.listId, ...draft.lists.items];
|
||||
});
|
||||
case LIST_EDITOR_REMOVE_SUCCESS:
|
||||
return state.updateIn(['lists', 'items'], list => (list as ImmutableList<string>).filterNot(item => item === action.listId));
|
||||
return create(state, (draft) => {
|
||||
draft.lists.items = draft.lists.items.filter(id => id !== action.listId);
|
||||
});
|
||||
default:
|
||||
return state;
|
||||
}
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import { List as ImmutableList, Record as ImmutableRecord } from 'immutable';
|
||||
import { create } from 'mutative';
|
||||
|
||||
import {
|
||||
LIST_CREATE_REQUEST,
|
||||
|
@ -22,83 +22,109 @@ import {
|
|||
|
||||
import type { AnyAction } from 'redux';
|
||||
|
||||
const AccountsRecord = ImmutableRecord({
|
||||
items: ImmutableList<string>(),
|
||||
loaded: false,
|
||||
isLoading: false,
|
||||
});
|
||||
interface State {
|
||||
listId: string | null;
|
||||
isSubmitting: boolean;
|
||||
isChanged: boolean;
|
||||
title: string;
|
||||
|
||||
const SuggestionsRecord = ImmutableRecord({
|
||||
value: '',
|
||||
items: ImmutableList<string>(),
|
||||
});
|
||||
accounts: {
|
||||
items: Array<string>;
|
||||
loaded: boolean;
|
||||
isLoading: boolean;
|
||||
};
|
||||
|
||||
const ReducerRecord = ImmutableRecord({
|
||||
listId: null as string | null,
|
||||
suggestions: {
|
||||
value: string;
|
||||
items: Array<string>;
|
||||
};
|
||||
}
|
||||
|
||||
const initialState: State = {
|
||||
listId: null,
|
||||
isSubmitting: false,
|
||||
isChanged: false,
|
||||
title: '',
|
||||
|
||||
accounts: AccountsRecord(),
|
||||
accounts: {
|
||||
items: [],
|
||||
loaded: false,
|
||||
isLoading: false,
|
||||
},
|
||||
|
||||
suggestions: SuggestionsRecord(),
|
||||
});
|
||||
suggestions: {
|
||||
value: '',
|
||||
items: [],
|
||||
},
|
||||
};
|
||||
|
||||
type State = ReturnType<typeof ReducerRecord>;
|
||||
|
||||
const listEditorReducer = (state: State = ReducerRecord(), action: AnyAction) => {
|
||||
const listEditorReducer = (state: State = initialState, action: AnyAction): State => {
|
||||
switch (action.type) {
|
||||
case LIST_EDITOR_RESET:
|
||||
return ReducerRecord();
|
||||
return initialState;
|
||||
case LIST_EDITOR_SETUP:
|
||||
return state.withMutations(map => {
|
||||
map.set('listId', action.list.id);
|
||||
map.set('title', action.list.title);
|
||||
map.set('isSubmitting', false);
|
||||
return create(state, (draft) => {
|
||||
draft.listId = action.list.id;
|
||||
draft.title = action.list.title;
|
||||
draft.isSubmitting = false;
|
||||
});
|
||||
case LIST_EDITOR_TITLE_CHANGE:
|
||||
return state.withMutations(map => {
|
||||
map.set('title', action.value);
|
||||
map.set('isChanged', true);
|
||||
return create(state, (draft) => {
|
||||
draft.title = action.value;
|
||||
draft.isChanged = true;
|
||||
});
|
||||
case LIST_CREATE_REQUEST:
|
||||
case LIST_UPDATE_REQUEST:
|
||||
return state.withMutations(map => {
|
||||
map.set('isSubmitting', true);
|
||||
map.set('isChanged', false);
|
||||
return create(state, (draft) => {
|
||||
draft.isSubmitting = true;
|
||||
draft.isChanged = false;
|
||||
});
|
||||
case LIST_CREATE_FAIL:
|
||||
case LIST_UPDATE_FAIL:
|
||||
return state.set('isSubmitting', false);
|
||||
return create(state, (draft) => {
|
||||
draft.isSubmitting = false;
|
||||
});
|
||||
case LIST_CREATE_SUCCESS:
|
||||
case LIST_UPDATE_SUCCESS:
|
||||
return state.withMutations(map => {
|
||||
map.set('isSubmitting', false);
|
||||
map.set('listId', action.list.id);
|
||||
return create(state, (draft) => {
|
||||
draft.isSubmitting = false;
|
||||
draft.listId = action.list.id;
|
||||
});
|
||||
case LIST_ACCOUNTS_FETCH_REQUEST:
|
||||
return state.setIn(['accounts', 'isLoading'], true);
|
||||
return create(state, (draft) => {
|
||||
draft.accounts.isLoading = true;
|
||||
});
|
||||
case LIST_ACCOUNTS_FETCH_FAIL:
|
||||
return state.setIn(['accounts', 'isLoading'], false);
|
||||
return create(state, (draft) => {
|
||||
draft.accounts.isLoading = false;
|
||||
});
|
||||
case LIST_ACCOUNTS_FETCH_SUCCESS:
|
||||
return state.update('accounts', accounts => accounts.withMutations(map => {
|
||||
map.set('isLoading', false);
|
||||
map.set('loaded', true);
|
||||
map.set('items', ImmutableList(action.accounts.map((item: { id: string }) => item.id)));
|
||||
}));
|
||||
return create(state, (draft) => {
|
||||
draft.accounts.isLoading = false;
|
||||
draft.accounts.loaded = true;
|
||||
draft.accounts.items = action.accounts.map((item: { id: string }) => item.id);
|
||||
});
|
||||
case LIST_EDITOR_SUGGESTIONS_CHANGE:
|
||||
return state.setIn(['suggestions', 'value'], action.value);
|
||||
return create(state, (draft) => {
|
||||
draft.suggestions.value = action.value;
|
||||
});
|
||||
case LIST_EDITOR_SUGGESTIONS_READY:
|
||||
return state.setIn(['suggestions', 'items'], ImmutableList(action.accounts.map((item: { id: string }) => item.id)));
|
||||
return create(state, (draft) => {
|
||||
draft.suggestions.items = action.accounts.map((item: { id: string }) => item.id);
|
||||
});
|
||||
case LIST_EDITOR_SUGGESTIONS_CLEAR:
|
||||
return state.update('suggestions', suggestions => suggestions.withMutations(map => {
|
||||
map.set('items', ImmutableList());
|
||||
map.set('value', '');
|
||||
}));
|
||||
return create(state, (draft) => {
|
||||
draft.suggestions.items = [];
|
||||
draft.suggestions.value = '';
|
||||
});
|
||||
case LIST_EDITOR_ADD_SUCCESS:
|
||||
return state.updateIn(['accounts', 'items'], list => (list as ImmutableList<string>).unshift(action.accountId));
|
||||
return create(state, (draft) => {
|
||||
draft.accounts.items = [action.accountId, ...draft.accounts.items];
|
||||
});
|
||||
case LIST_EDITOR_REMOVE_SUCCESS:
|
||||
return state.updateIn(['accounts', 'items'], list => (list as ImmutableList<string>).filterNot((item) => item === action.accountId));
|
||||
return create(state, (draft) => {
|
||||
draft.accounts.items = draft.accounts.items.filter(id => id !== action.accoundId);
|
||||
});
|
||||
default:
|
||||
return state;
|
||||
}
|
||||
|
|
|
@ -1,4 +1,3 @@
|
|||
import { OrderedSet as ImmutableOrderedSet } from 'immutable';
|
||||
import { createSelector } from 'reselect';
|
||||
|
||||
// import { getLocale } from 'pl-fe/actions/settings';
|
||||
|
@ -262,14 +261,11 @@ const makeGetReport = () => {
|
|||
|
||||
const getAuthUserIds = createSelector(
|
||||
[(state: RootState) => state.auth.users],
|
||||
authUsers => authUsers.reduce((userIds: ImmutableOrderedSet<string>, authUser) => {
|
||||
try {
|
||||
const userId = authUser.id;
|
||||
return validId(userId) ? userIds.add(userId) : userIds;
|
||||
} catch {
|
||||
authUsers => authUsers.reduce((userIds: Array<string>, authUser) => {
|
||||
const userId = authUser?.id;
|
||||
if (validId(userId)) userIds.push(userId);
|
||||
return userIds;
|
||||
}
|
||||
}, ImmutableOrderedSet<string>()));
|
||||
}, []));
|
||||
|
||||
const makeGetOtherAccounts = () => createSelector([
|
||||
(state: RootState) => state.entities[Entities.ACCOUNTS]?.store as EntityStore<Account>,
|
||||
|
@ -314,9 +310,7 @@ const makeGetHosts = () =>
|
|||
createSelector([getSimplePolicy], (simplePolicy) => {
|
||||
const { accept, reject_deletes, report_removal, ...rest } = simplePolicy;
|
||||
|
||||
return Object.values(rest)
|
||||
.reduce((acc, hosts) => acc.union(hosts), ImmutableOrderedSet())
|
||||
.sort();
|
||||
return [...new Set(Object.values(rest).reduce((acc, hosts) => (acc.push(...hosts), acc), []))].toSorted();
|
||||
});
|
||||
|
||||
interface RemoteInstance {
|
||||
|
|
|
@ -1,5 +1,3 @@
|
|||
import { List as ImmutableList } from 'immutable';
|
||||
|
||||
import { selectAccount, selectOwnAccount } from 'pl-fe/selectors';
|
||||
|
||||
import type { RootState } from 'pl-fe/store';
|
||||
|
@ -43,19 +41,19 @@ const getAccessToken = (state: RootState) => {
|
|||
const getAuthUserId = (state: RootState) => {
|
||||
const me = state.auth.me;
|
||||
|
||||
return ImmutableList([
|
||||
return [
|
||||
state.auth.users.get(me!)?.id,
|
||||
me,
|
||||
].filter(id => id)).find(validId);
|
||||
].filter(id => id).find(validId);
|
||||
};
|
||||
|
||||
const getAuthUserUrl = (state: RootState) => {
|
||||
const me = state.auth.me;
|
||||
|
||||
return ImmutableList([
|
||||
return [
|
||||
state.auth.users.get(me!)?.url,
|
||||
me,
|
||||
].filter(url => url)).find(isURL);
|
||||
].filter(url => url).find(isURL);
|
||||
};
|
||||
|
||||
/** Get the VAPID public key. */
|
||||
|
|
|
@ -1,5 +1,3 @@
|
|||
import { OrderedSet as ImmutableOrderedSet } from 'immutable';
|
||||
|
||||
import type { Account } from 'pl-fe/normalizers/account';
|
||||
|
||||
/** Convert a plain tag into a badge. */
|
||||
|
@ -17,15 +15,10 @@ interface TagDiff {
|
|||
}
|
||||
|
||||
/** Returns the differences between two sets of tags. */
|
||||
const getTagDiff = (oldTags: string[], newTags: string[]): TagDiff => {
|
||||
const o = ImmutableOrderedSet(oldTags);
|
||||
const n = ImmutableOrderedSet(newTags);
|
||||
|
||||
return {
|
||||
added: n.subtract(o).toArray(),
|
||||
removed: o.subtract(n).toArray(),
|
||||
};
|
||||
};
|
||||
const getTagDiff = (oldTags: string[], newTags: string[]): TagDiff => ({
|
||||
added: newTags.filter(tag => !oldTags.includes(tag)),
|
||||
removed: oldTags.filter(tag => !newTags.includes(tag)),
|
||||
});
|
||||
|
||||
/** Returns only tags which are badges. */
|
||||
const filterBadges = (tags: string[]): string[] => tags.filter(tag => tag.startsWith('badge:'));
|
||||
|
|
Loading…
Reference in a new issue