import { pipe } from 'ramda'; import { useEffect, useState } from 'react'; import { useLocation, useParams } from 'react-router-dom'; import { Card } from 'reactstrap'; import type { ShlinkShortUrlsListParams, ShlinkShortUrlsOrder } from '../../api/types'; import { boundToMercureHub } from '../../mercure/helpers/boundToMercureHub'; import { Topics } from '../../mercure/helpers/Topics'; import type { SelectedServer } from '../../servers/data'; import { getServerId } from '../../servers/data'; import type { Settings } from '../../settings/reducers/settings'; import { DEFAULT_SHORT_URLS_ORDERING } from '../../settings/reducers/settings'; import { useFeature } from '../../utils/helpers/features'; import type { OrderDir } from '../../utils/helpers/ordering'; import { determineOrderDir } from '../../utils/helpers/ordering'; import { TableOrderIcon } from '../../utils/table/TableOrderIcon'; import type { ShortUrlsOrder, ShortUrlsOrderableFields } from './data'; import { useShortUrlsQuery } from './helpers/hooks'; import { Paginator } from './Paginator'; import type { ShortUrlsList as ShortUrlsListState } from './reducers/shortUrlsList'; import type { ShortUrlsFilteringBarType } from './ShortUrlsFilteringBar'; import type { ShortUrlsTableType } from './ShortUrlsTable'; interface ShortUrlsListProps { selectedServer: SelectedServer; shortUrlsList: ShortUrlsListState; listShortUrls: (params: ShlinkShortUrlsListParams) => void; settings: Settings; } export const ShortUrlsList = ( ShortUrlsTable: ShortUrlsTableType, ShortUrlsFilteringBar: ShortUrlsFilteringBarType, ) => boundToMercureHub(({ listShortUrls, shortUrlsList, selectedServer, settings }: ShortUrlsListProps) => { const serverId = getServerId(selectedServer); const { page } = useParams(); const location = useLocation(); const [filter, toFirstPage] = useShortUrlsQuery(); const { tags, search, startDate, endDate, orderBy, tagsMode, excludeBots, excludePastValidUntil, excludeMaxVisitsReached, } = filter; const [actualOrderBy, setActualOrderBy] = useState( // This separated state handling is needed to be able to fall back to settings value, but only once when loaded orderBy ?? settings.shortUrlsList?.defaultOrdering ?? DEFAULT_SHORT_URLS_ORDERING, ); const { pagination } = shortUrlsList?.shortUrls ?? {}; const doExcludeBots = excludeBots ?? settings.visits?.excludeBots; const supportsExcludingBots = useFeature('excludeBotsOnShortUrls', selectedServer); const handleOrderBy = (field?: ShortUrlsOrderableFields, dir?: OrderDir) => { toFirstPage({ orderBy: { field, dir } }); setActualOrderBy({ field, dir }); }; const orderByColumn = (field: ShortUrlsOrderableFields) => () => handleOrderBy(field, determineOrderDir(field, actualOrderBy.field, actualOrderBy.dir)); const renderOrderIcon = (field: ShortUrlsOrderableFields) => ; const addTag = pipe( (newTag: string) => [...new Set([...tags, newTag])], (updatedTags) => toFirstPage({ tags: updatedTags }), ); const parseOrderByForShlink = ({ field, dir }: ShortUrlsOrder): ShlinkShortUrlsOrder => { if (supportsExcludingBots && doExcludeBots && field === 'visits') { return { field: 'nonBotVisits', dir }; } return { field, dir }; }; useEffect(() => { listShortUrls({ page, searchTerm: search, tags, startDate, endDate, orderBy: parseOrderByForShlink(actualOrderBy), tagsMode, excludePastValidUntil, excludeMaxVisitsReached, }); }, [ page, search, tags, startDate, endDate, actualOrderBy.field, actualOrderBy.dir, tagsMode, excludePastValidUntil, excludeMaxVisitsReached, ]); return ( <> ); }, () => [Topics.visits]);