/* eslint-disable lines-around-comment */ import { getURLWithoutParamsNormalized } from '../base/connection/utils'; import PersistenceRegistry from '../base/redux/PersistenceRegistry'; import ReducerRegistry from '../base/redux/ReducerRegistry'; import { _STORE_CURRENT_CONFERENCE, _UPDATE_CONFERENCE_DURATION, DELETE_RECENT_LIST_ENTRY } from './actionTypes'; // @ts-ignore import { isRecentListEnabled } from './functions'; interface IRecent { conference: string; date: number; duration: number; } export type IRecentListState = IRecent[]; /** * The default/initial redux state of the feature {@code recent-list}. * * @type {IRecentListState} */ const DEFAULT_STATE: IRecentListState = []; /** * The max size of the list. * * @type {number} */ export const MAX_LIST_SIZE = 30; /** * The redux subtree of this feature. */ const STORE_NAME = 'features/recent-list'; /** * Sets up the persistence of the feature {@code recent-list}. */ PersistenceRegistry.register(STORE_NAME); /** * Reduces redux actions for the purposes of the feature {@code recent-list}. */ ReducerRegistry.register(STORE_NAME, (state = DEFAULT_STATE, action): IRecentListState => { if (isRecentListEnabled()) { switch (action.type) { case DELETE_RECENT_LIST_ENTRY: return _deleteRecentListEntry(state, action.entryId); case _STORE_CURRENT_CONFERENCE: return _storeCurrentConference(state, action); case _UPDATE_CONFERENCE_DURATION: return _updateConferenceDuration(state, action); default: return state; } } return state; }); /** * Deletes a recent list entry based on the url and date of the item. * * @param {IRecentListState} state - The Redux state. * @param {Object} entryId - The ID object of the entry. * @returns {IRecentListState} */ function _deleteRecentListEntry( state: Array, entryId: { date: number; url: string; }): Array { return state.filter(entry => entry.conference !== entryId.url || entry.date !== entryId.date); } /** * Adds a new list entry to the redux store. * * @param {IRecentListState} state - The redux state of the feature {@code recent-list}. * @param {Object} action - The redux action. * @returns {Object} */ function _storeCurrentConference(state: IRecentListState, { locationURL }: { locationURL: { href: string; }; }) { const conference = locationURL.href; // If the current conference is already in the list, we remove it to re-add // it to the top. const nextState = state.filter(e => !_urlStringEquals(e.conference, conference)); // The list is a reverse-sorted (i.e. the newer elements are at the end). nextState.push({ conference, date: Date.now(), duration: 0 // We don't have the duration yet! }); // Ensure the list doesn't exceed a/the maximum size. nextState.splice(0, nextState.length - MAX_LIST_SIZE); return nextState; } /** * Updates the conference length when left. * * @param {IRecentListState} state - The redux state of the feature {@code recent-list}. * @param {Object} action - The redux action. * @returns {Object} The next redux state of the feature {@code recent-list}. */ function _updateConferenceDuration(state: IRecentListState, { locationURL }: { locationURL: { href: string; }; }) { if (locationURL?.href && state.length) { const mostRecentIndex = state.length - 1; const mostRecent = state[mostRecentIndex]; if (_urlStringEquals(mostRecent.conference, locationURL.href)) { // The last conference start was stored so we need to update the // length. const nextMostRecent = { ...mostRecent, duration: Date.now() - mostRecent.date }; // Shallow copy to avoid in-place modification. const nextState = state.slice(); nextState[mostRecentIndex] = nextMostRecent; return nextState; } } return state; } /** * Determines whether two specific URL {@code strings} are equal in the sense * that they identify one and the same conference resource (irrespective of * time) for the purposes of the feature {@code recent-list}. * * @param {string} a - The URL {@code string} to test for equality to {@code b}. * @param {string} b - The URL {@code string} to test for equality to {@code a}. * @returns {boolean} */ function _urlStringEquals(a: string, b: string) { const aHref = getURLWithoutParamsNormalized(new URL(a)); const bHref = getURLWithoutParamsNormalized(new URL(b)); return aHref === bHref; }