frontend-rw #1

Merged
marcin merged 347 commits from frontend-rw into develop 2024-12-05 15:32:18 -08:00
3 changed files with 45 additions and 101 deletions
Showing only changes of commit 74d0d4c60b - Show all commits

View file

@ -24,7 +24,6 @@ import type { AppDispatch, RootState } from 'pl-fe/store';
const NOTIFICATIONS_UPDATE = 'NOTIFICATIONS_UPDATE' as const; const NOTIFICATIONS_UPDATE = 'NOTIFICATIONS_UPDATE' as const;
const NOTIFICATIONS_UPDATE_NOOP = 'NOTIFICATIONS_UPDATE_NOOP' as const; const NOTIFICATIONS_UPDATE_NOOP = 'NOTIFICATIONS_UPDATE_NOOP' as const;
const NOTIFICATIONS_UPDATE_QUEUE = 'NOTIFICATIONS_UPDATE_QUEUE' as const; const NOTIFICATIONS_UPDATE_QUEUE = 'NOTIFICATIONS_UPDATE_QUEUE' as const;
const NOTIFICATIONS_DEQUEUE = 'NOTIFICATIONS_DEQUEUE' as const;
const NOTIFICATIONS_EXPAND_REQUEST = 'NOTIFICATIONS_EXPAND_REQUEST' as const; const NOTIFICATIONS_EXPAND_REQUEST = 'NOTIFICATIONS_EXPAND_REQUEST' as const;
const NOTIFICATIONS_EXPAND_SUCCESS = 'NOTIFICATIONS_EXPAND_SUCCESS' as const; const NOTIFICATIONS_EXPAND_SUCCESS = 'NOTIFICATIONS_EXPAND_SUCCESS' as const;
@ -100,8 +99,6 @@ const updateNotificationsQueue = (notification: BaseNotification, intlMessages:
let filtered: boolean | null = false; let filtered: boolean | null = false;
const isOnNotificationsPage = curPath === '/notifications';
if (notification.type === 'mention' || notification.type === 'status') { if (notification.type === 'mention' || notification.type === 'status') {
const regex = regexFromFilters(filters); const regex = regexFromFilters(filters);
const searchIndex = notification.status.spoiler_text + '\n' + unescapeHTML(notification.status.content); const searchIndex = notification.status.spoiler_text + '\n' + unescapeHTML(notification.status.content);
@ -139,37 +136,7 @@ const updateNotificationsQueue = (notification: BaseNotification, intlMessages:
}); });
} }
if (isOnNotificationsPage) {
dispatch({
type: NOTIFICATIONS_UPDATE_QUEUE,
notification,
intlMessages,
intlLocale,
});
} else {
dispatch(updateNotifications(notification)); dispatch(updateNotifications(notification));
}
};
const dequeueNotifications = () =>
(dispatch: AppDispatch, getState: () => RootState) => {
const queuedNotifications = getState().notifications.queuedNotifications;
const totalQueuedNotificationsCount = getState().notifications.totalQueuedNotificationsCount;
if (totalQueuedNotificationsCount === 0) {
return;
} else if (totalQueuedNotificationsCount > 0 && totalQueuedNotificationsCount <= MAX_QUEUED_NOTIFICATIONS) {
queuedNotifications.forEach((block) => {
dispatch(updateNotifications(block.notification));
});
} else {
dispatch(expandNotifications());
}
dispatch({
type: NOTIFICATIONS_DEQUEUE,
});
dispatch(markReadNotifications());
}; };
const excludeTypesFromFilter = (filters: string[]) => NOTIFICATION_TYPES.filter(item => !filters.includes(item)); const excludeTypesFromFilter = (filters: string[]) => NOTIFICATION_TYPES.filter(item => !filters.includes(item));
@ -295,7 +262,6 @@ export {
NOTIFICATIONS_UPDATE, NOTIFICATIONS_UPDATE,
NOTIFICATIONS_UPDATE_NOOP, NOTIFICATIONS_UPDATE_NOOP,
NOTIFICATIONS_UPDATE_QUEUE, NOTIFICATIONS_UPDATE_QUEUE,
NOTIFICATIONS_DEQUEUE,
NOTIFICATIONS_EXPAND_REQUEST, NOTIFICATIONS_EXPAND_REQUEST,
NOTIFICATIONS_EXPAND_SUCCESS, NOTIFICATIONS_EXPAND_SUCCESS,
NOTIFICATIONS_EXPAND_FAIL, NOTIFICATIONS_EXPAND_FAIL,
@ -309,7 +275,6 @@ export {
type FilterType, type FilterType,
updateNotifications, updateNotifications,
updateNotificationsQueue, updateNotificationsQueue,
dequeueNotifications,
expandNotifications, expandNotifications,
expandNotificationsRequest, expandNotificationsRequest,
expandNotificationsSuccess, expandNotificationsSuccess,

View file

@ -1,13 +1,13 @@
import clsx from 'clsx'; import clsx from 'clsx';
import debounce from 'lodash/debounce'; import debounce from 'lodash/debounce';
import React, { useCallback, useEffect, useRef } from 'react'; import React, { useCallback, useEffect, useRef, useState } from 'react';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl'; import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
import { createSelector } from 'reselect'; import { createSelector } from 'reselect';
import { import {
expandNotifications, expandNotifications,
markReadNotifications,
scrollTopNotifications, scrollTopNotifications,
dequeueNotifications,
} from 'pl-fe/actions/notifications'; } from 'pl-fe/actions/notifications';
import PullToRefresh from 'pl-fe/components/pull-to-refresh'; import PullToRefresh from 'pl-fe/components/pull-to-refresh';
import ScrollTopButton from 'pl-fe/components/scroll-top-button'; import ScrollTopButton from 'pl-fe/components/scroll-top-button';
@ -31,7 +31,27 @@ const messages = defineMessages({
const getNotifications = createSelector([ const getNotifications = createSelector([
(state: RootState) => state.notifications.items.toArray(), (state: RootState) => state.notifications.items.toArray(),
], (notifications) => notifications.map(([_, notification]) => notification).filter(item => item !== null)); (_, topNotification?: string) => topNotification,
], (notifications, topNotificationId) => {
const allNotifications = notifications.map(([_, notification]) => notification).filter(item => item !== null);
if (topNotificationId) {
const queuedNotificationCount = allNotifications.findIndex((notification) =>
notification.most_recent_notification_id <= topNotificationId,
);
const displayedNotifications = allNotifications.slice(queuedNotificationCount);
return {
queuedNotificationCount,
displayedNotifications,
};
}
return {
queuedNotificationCount: 0,
displayedNotifications: allNotifications,
};
});
const Notifications = () => { const Notifications = () => {
const dispatch = useAppDispatch(); const dispatch = useAppDispatch();
@ -40,11 +60,11 @@ const Notifications = () => {
const showFilterBar = settings.notifications.quickFilter.show; const showFilterBar = settings.notifications.quickFilter.show;
const activeFilter = settings.notifications.quickFilter.active; const activeFilter = settings.notifications.quickFilter.active;
const notifications = useAppSelector(state => getNotifications(state)); const [topNotification, setTopNotification] = useState<string>();
const { queuedNotificationCount, displayedNotifications } = useAppSelector(state => getNotifications(state, topNotification));
const isLoading = useAppSelector(state => state.notifications.isLoading); const isLoading = useAppSelector(state => state.notifications.isLoading);
// const isUnread = useAppSelector(state => state.notifications.unread > 0); // const isUnread = useAppSelector(state => state.notifications.unread > 0);
const hasMore = useAppSelector(state => state.notifications.hasMore); const hasMore = useAppSelector(state => state.notifications.hasMore);
const totalQueuedNotificationsCount = useAppSelector(state => state.notifications.totalQueuedNotificationsCount || 0);
const scrollableContentRef = useRef<Array<JSX.Element> | null>(null); const scrollableContentRef = useRef<Array<JSX.Element> | null>(null);
@ -53,26 +73,26 @@ const Notifications = () => {
// }; // };
const handleLoadOlder = useCallback(debounce(() => { const handleLoadOlder = useCallback(debounce(() => {
const minId = notifications.reduce<string | undefined>( const minId = displayedNotifications.reduce<string | undefined>(
(minId, notification) => minId && notification.page_min_id && notification.page_min_id > minId (minId, notification) => minId && notification.page_min_id && notification.page_min_id > minId
? minId ? minId
: notification.page_min_id, : notification.page_min_id,
undefined, undefined,
); );
dispatch(expandNotifications({ maxId: minId })); dispatch(expandNotifications({ maxId: minId }));
}, 300, { leading: true }), [notifications]); }, 300, { leading: true }), [displayedNotifications]);
const handleScroll = useCallback(debounce((startIndex?: number) => { const handleScroll = useCallback(debounce((startIndex?: number) => {
dispatch(scrollTopNotifications(startIndex === 0)); dispatch(scrollTopNotifications(startIndex === 0));
}, 100), []); }, 100), []);
const handleMoveUp = (id: string) => { const handleMoveUp = (id: string) => {
const elementIndex = notifications.findIndex(item => item !== null && item.group_key === id) - 1; const elementIndex = displayedNotifications.findIndex(item => item !== null && item.group_key === id) - 1;
_selectChild(elementIndex); _selectChild(elementIndex);
}; };
const handleMoveDown = (id: string) => { const handleMoveDown = (id: string) => {
const elementIndex = notifications.findIndex(item => item !== null && item.group_key === id) + 1; const elementIndex = displayedNotifications.findIndex(item => item !== null && item.group_key === id) + 1;
_selectChild(elementIndex); _selectChild(elementIndex);
}; };
@ -84,7 +104,8 @@ const Notifications = () => {
}; };
const handleDequeueNotifications = useCallback(() => { const handleDequeueNotifications = useCallback(() => {
dispatch(dequeueNotifications()); setTopNotification(undefined);
dispatch(markReadNotifications());
}, []); }, []);
const handleRefresh = useCallback(() => dispatch(expandNotifications()), []); const handleRefresh = useCallback(() => dispatch(expandNotifications()), []);
@ -100,6 +121,11 @@ const Notifications = () => {
}; };
}, []); }, []);
useEffect(() => {
if (topNotification || displayedNotifications.length === 0) return;
setTopNotification(displayedNotifications[0].most_recent_notification_id);
}, [displayedNotifications.length]);
const emptyMessage = activeFilter === 'all' const emptyMessage = activeFilter === 'all'
? <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' 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." />; : <FormattedMessage id='empty_column.notifications_filtered' defaultMessage="You don't have any notifications of this type yet." />;
@ -112,8 +138,8 @@ const Notifications = () => {
if (isLoading && scrollableContentRef.current) { if (isLoading && scrollableContentRef.current) {
scrollableContent = scrollableContentRef.current; scrollableContent = scrollableContentRef.current;
} else if (notifications.length > 0 || hasMore) { } else if (displayedNotifications.length > 0 || hasMore) {
scrollableContent = notifications.map((item) => ( scrollableContent = displayedNotifications.map((item) => (
<Notification <Notification
key={item.group_key} key={item.group_key}
notification={item} notification={item}
@ -130,7 +156,7 @@ const Notifications = () => {
const scrollContainer = ( const scrollContainer = (
<ScrollableList <ScrollableList
isLoading={isLoading} isLoading={isLoading}
showLoading={isLoading && notifications.length === 0} showLoading={isLoading && displayedNotifications.length === 0}
hasMore={hasMore} hasMore={hasMore}
emptyMessage={emptyMessage} emptyMessage={emptyMessage}
placeholderComponent={PlaceholderNotification} placeholderComponent={PlaceholderNotification}
@ -138,7 +164,7 @@ const Notifications = () => {
onLoadMore={handleLoadOlder} onLoadMore={handleLoadOlder}
onScroll={handleScroll} onScroll={handleScroll}
listClassName={clsx('divide-y divide-solid divide-gray-200 black:divide-gray-800 dark:divide-primary-800', { listClassName={clsx('divide-y divide-solid divide-gray-200 black:divide-gray-800 dark:divide-primary-800', {
'animate-pulse': notifications.length === 0, 'animate-pulse': displayedNotifications.length === 0,
})} })}
> >
{scrollableContent!} {scrollableContent!}
@ -152,7 +178,7 @@ const Notifications = () => {
<Portal> <Portal>
<ScrollTopButton <ScrollTopButton
onClick={handleDequeueNotifications} onClick={handleDequeueNotifications}
count={totalQueuedNotificationsCount} count={queuedNotificationCount}
message={messages.queue} message={messages.queue}
/> />
</Portal> </Portal>

View file

@ -20,41 +20,29 @@ import {
NOTIFICATIONS_FILTER_SET, NOTIFICATIONS_FILTER_SET,
NOTIFICATIONS_CLEAR, NOTIFICATIONS_CLEAR,
NOTIFICATIONS_SCROLL_TOP, NOTIFICATIONS_SCROLL_TOP,
NOTIFICATIONS_UPDATE_QUEUE,
NOTIFICATIONS_DEQUEUE,
NOTIFICATIONS_MARK_READ_REQUEST, NOTIFICATIONS_MARK_READ_REQUEST,
MAX_QUEUED_NOTIFICATIONS,
} from '../actions/notifications'; } from '../actions/notifications';
import { TIMELINE_DELETE, type TimelineAction } from '../actions/timelines'; import { TIMELINE_DELETE, type TimelineAction } from '../actions/timelines';
import type { Notification as BaseNotification, Markers, NotificationGroup, PaginatedResponse, Relationship } from 'pl-api'; import type { Notification as BaseNotification, Markers, NotificationGroup, PaginatedResponse, Relationship } from 'pl-api';
import type { AnyAction } from 'redux'; import type { AnyAction } from 'redux';
const QueuedNotificationRecord = ImmutableRecord({
notification: {} as any as BaseNotification,
intlMessages: {} as Record<string, string>,
intlLocale: '',
});
const ReducerRecord = ImmutableRecord({ const ReducerRecord = ImmutableRecord({
items: ImmutableOrderedMap<string, NotificationGroup>(), items: ImmutableOrderedMap<string, NotificationGroup>(),
hasMore: true, hasMore: true,
top: false, top: false,
unread: 0, unread: 0,
isLoading: false, isLoading: false,
queuedNotifications: ImmutableOrderedMap<string, QueuedNotification>(), //max = MAX_QUEUED_NOTIFICATIONS
totalQueuedNotificationsCount: 0, //used for queuedItems overflow for MAX_QUEUED_NOTIFICATIONS+
lastRead: -1 as string | -1, lastRead: -1 as string | -1,
}); });
type State = ReturnType<typeof ReducerRecord>; type State = ReturnType<typeof ReducerRecord>;
type QueuedNotification = ReturnType<typeof QueuedNotificationRecord>;
const parseId = (id: string | number) => parseInt(id as string, 10); const parseId = (id: string | number) => parseInt(id as string, 10);
// For sorting the notifications // For sorting the notifications
const comparator = (a: Pick<NotificationGroup, 'group_key'>, b: Pick<NotificationGroup, 'group_key'>) => { const comparator = (a: Pick<NotificationGroup, 'most_recent_notification_id'>, b: Pick<NotificationGroup, 'most_recent_notification_id'>) => {
const parse = (m: Pick<NotificationGroup, 'group_key'>) => parseId(m.group_key); const parse = (m: Pick<NotificationGroup, 'most_recent_notification_id'>) => parseId(m.most_recent_notification_id);
if (parse(a) < parse(b)) return 1; if (parse(a) < parse(b)) return 1;
if (parse(a) > parse(b)) return -1; if (parse(a) > parse(b)) return -1;
return 0; return 0;
@ -75,13 +63,7 @@ const importNotification = (state: State, notification: NotificationGroup) => {
if (!top) state = state.update('unread', unread => unread + 1); if (!top) state = state.update('unread', unread => unread + 1);
return state.update('items', map => { return state.update('items', map => map.set(notification.group_key, notification).sort(comparator));
if (top && map.size > 40) {
map = map.take(20);
}
return map.set(notification.group_key, notification).sort(comparator);
});
}; };
const expandNormalizedNotifications = (state: State, notifications: NotificationGroup[], next: (() => Promise<PaginatedResponse<BaseNotification>>) | null) => { const expandNormalizedNotifications = (state: State, notifications: NotificationGroup[], next: (() => Promise<PaginatedResponse<BaseNotification>>) | null) => {
@ -112,28 +94,6 @@ const deleteByStatus = (state: State, statusId: string) =>
// @ts-ignore // @ts-ignore
state.update('items', map => map.filterNot(item => item !== null && item.status === statusId)); state.update('items', map => map.filterNot(item => item !== null && item.status === statusId));
const updateNotificationsQueue = (state: State, notification: BaseNotification, intlMessages: Record<string, string>, intlLocale: string) => {
const queuedNotifications = state.queuedNotifications;
const listedNotifications = state.items;
const totalQueuedNotificationsCount = state.totalQueuedNotificationsCount;
const alreadyExists = queuedNotifications.has(notification.group_key) || listedNotifications.has(notification.group_key);
if (alreadyExists) return state;
const newQueuedNotifications = queuedNotifications;
return state.withMutations(mutable => {
if (totalQueuedNotificationsCount <= MAX_QUEUED_NOTIFICATIONS) {
mutable.set('queuedNotifications', newQueuedNotifications.set(notification.group_key, QueuedNotificationRecord({
notification,
intlMessages,
intlLocale,
})));
}
mutable.set('totalQueuedNotificationsCount', totalQueuedNotificationsCount + 1);
});
};
const importMarker = (state: State, marker: Markers) => { const importMarker = (state: State, marker: Markers) => {
const lastReadId = marker.notifications.last_read_id || -1 as string | -1; const lastReadId = marker.notifications.last_read_id || -1 as string | -1;
@ -163,13 +123,6 @@ const notifications = (state: State = ReducerRecord(), action: AccountsAction |
return updateTop(state, action.top); return updateTop(state, action.top);
case NOTIFICATIONS_UPDATE: case NOTIFICATIONS_UPDATE:
return importNotification(state, action.notification); return importNotification(state, action.notification);
case NOTIFICATIONS_UPDATE_QUEUE:
return updateNotificationsQueue(state, action.notification, action.intlMessages, action.intlLocale);
case NOTIFICATIONS_DEQUEUE:
return state.withMutations(mutable => {
mutable.delete('queuedNotifications');
mutable.set('totalQueuedNotificationsCount', 0);
});
case NOTIFICATIONS_EXPAND_SUCCESS: case NOTIFICATIONS_EXPAND_SUCCESS:
return expandNormalizedNotifications(state, action.notifications, action.next); return expandNormalizedNotifications(state, action.notifications, action.next);
case ACCOUNT_BLOCK_SUCCESS: case ACCOUNT_BLOCK_SUCCESS: