import type { IDatasource } from '@ag-grid-community/core'; import type { AgGridReact as AgGridReactType } from '@ag-grid-community/react/lib/agGridReact'; import { Flex, Group, Stack } from '@mantine/core'; import debounce from 'lodash/debounce'; import { ChangeEvent, MouseEvent, MutableRefObject, useCallback } from 'react'; import { RiArrowDownSLine, RiFilter3Line, RiFolder2Line, RiMoreFill, RiSortAsc, RiSortDesc, } from 'react-icons/ri'; import { api } from '/@/renderer/api'; import { queryKeys } from '/@/renderer/api/query-keys'; import { ServerType, SongListSort, SortOrder } from '/@/renderer/api/types'; import { Button, DropdownMenu, PageHeader, SearchInput, Slider, TextTitle, Switch, MultiSelect, Text, SONG_TABLE_COLUMNS, } from '/@/renderer/components'; import { useMusicFolders } from '/@/renderer/features/shared'; import { JellyfinSongFilters } from '/@/renderer/features/songs/components/jellyfin-song-filters'; import { NavidromeSongFilters } from '/@/renderer/features/songs/components/navidrome-song-filters'; import { useContainerQuery } from '/@/renderer/hooks'; import { queryClient } from '/@/renderer/lib/react-query'; import { SongListFilter, useCurrentServer, useSetSongFilters, useSetSongStore, useSetSongTable, useSetSongTablePagination, useSongListStore, } from '/@/renderer/store'; import { ListDisplayType, TableColumn } from '/@/renderer/types'; const FILTERS = { jellyfin: [ { defaultOrder: SortOrder.ASC, name: 'Album', value: SongListSort.ALBUM }, { defaultOrder: SortOrder.ASC, name: 'Album Artist', value: SongListSort.ALBUM_ARTIST }, { defaultOrder: SortOrder.ASC, name: 'Artist', value: SongListSort.ARTIST }, { defaultOrder: SortOrder.ASC, name: 'Duration', value: SongListSort.DURATION }, { defaultOrder: SortOrder.ASC, name: 'Most Played', value: SongListSort.PLAY_COUNT }, { defaultOrder: SortOrder.ASC, name: 'Name', value: SongListSort.NAME }, { defaultOrder: SortOrder.ASC, name: 'Random', value: SongListSort.RANDOM }, { defaultOrder: SortOrder.ASC, name: 'Recently Added', value: SongListSort.RECENTLY_ADDED }, { defaultOrder: SortOrder.ASC, name: 'Recently Played', value: SongListSort.RECENTLY_PLAYED }, { defaultOrder: SortOrder.ASC, name: 'Release Date', value: SongListSort.RELEASE_DATE }, ], navidrome: [ { defaultOrder: SortOrder.ASC, name: 'Album', value: SongListSort.ALBUM }, { defaultOrder: SortOrder.ASC, name: 'Album Artist', value: SongListSort.ALBUM_ARTIST }, { defaultOrder: SortOrder.ASC, name: 'Artist', value: SongListSort.ARTIST }, { defaultOrder: SortOrder.DESC, name: 'BPM', value: SongListSort.BPM }, { defaultOrder: SortOrder.ASC, name: 'Channels', value: SongListSort.CHANNELS }, { defaultOrder: SortOrder.ASC, name: 'Comment', value: SongListSort.COMMENT }, { defaultOrder: SortOrder.DESC, name: 'Duration', value: SongListSort.DURATION }, { defaultOrder: SortOrder.DESC, name: 'Favorited', value: SongListSort.FAVORITED }, { defaultOrder: SortOrder.ASC, name: 'Genre', value: SongListSort.GENRE }, { defaultOrder: SortOrder.ASC, name: 'Name', value: SongListSort.NAME }, { defaultOrder: SortOrder.DESC, name: 'Play Count', value: SongListSort.PLAY_COUNT }, { defaultOrder: SortOrder.DESC, name: 'Rating', value: SongListSort.RATING }, { defaultOrder: SortOrder.DESC, name: 'Recently Added', value: SongListSort.RECENTLY_ADDED }, { defaultOrder: SortOrder.DESC, name: 'Recently Played', value: SongListSort.RECENTLY_PLAYED }, { defaultOrder: SortOrder.DESC, name: 'Year', value: SongListSort.YEAR }, ], }; const ORDER = [ { name: 'Ascending', value: SortOrder.ASC }, { name: 'Descending', value: SortOrder.DESC }, ]; interface SongListHeaderProps { tableRef: MutableRefObject; } export const SongListHeader = ({ tableRef }: SongListHeaderProps) => { const server = useCurrentServer(); const page = useSongListStore(); const setPage = useSetSongStore(); const setFilter = useSetSongFilters(); const setTable = useSetSongTable(); const setPagination = useSetSongTablePagination(); const cq = useContainerQuery(); const musicFoldersQuery = useMusicFolders(); const sortByLabel = (server?.type && (FILTERS[server.type as keyof typeof FILTERS] as { name: string; value: string }[]).find( (f) => f.value === page.filter.sortBy, )?.name) || 'Unknown'; const sortOrderLabel = ORDER.find((s) => s.value === page.filter.sortOrder)?.name; const handleFilterChange = useCallback( async (filters?: SongListFilter) => { const dataSource: IDatasource = { getRows: async (params) => { const limit = params.endRow - params.startRow; const startIndex = params.startRow; const pageFilters = filters || page.filter; const queryKey = queryKeys.songs.list(server?.id || '', { limit, startIndex, ...pageFilters, }); const songsRes = await queryClient.fetchQuery(queryKey, async ({ signal }) => api.controller.getSongList({ query: { limit, startIndex, ...pageFilters, }, server, signal, }), ); const songs = api.normalize.songList(songsRes, server); params.successCallback(songs?.items || [], songsRes?.totalRecordCount); }, rowCount: undefined, }; tableRef.current?.api.setDatasource(dataSource); tableRef.current?.api.purgeInfiniteCache(); tableRef.current?.api.ensureIndexVisible(0, 'top'); setPagination({ currentPage: 0 }); }, [page.filter, server, setPagination, tableRef], ); const handleSetSortBy = useCallback( (e: MouseEvent) => { if (!e.currentTarget?.value || !server?.type) return; const sortOrder = FILTERS[server.type as keyof typeof FILTERS].find( (f) => f.value === e.currentTarget.value, )?.defaultOrder; const updatedFilters = setFilter({ sortBy: e.currentTarget.value as SongListSort, sortOrder: sortOrder || SortOrder.ASC, }); handleFilterChange(updatedFilters); }, [handleFilterChange, server?.type, setFilter], ); const handleSetMusicFolder = useCallback( (e: MouseEvent) => { if (!e.currentTarget?.value) return; let updatedFilters = null; if (e.currentTarget.value === String(page.filter.musicFolderId)) { updatedFilters = setFilter({ musicFolderId: undefined }); } else { updatedFilters = setFilter({ musicFolderId: e.currentTarget.value }); } handleFilterChange(updatedFilters); }, [handleFilterChange, page.filter.musicFolderId, setFilter], ); const handleToggleSortOrder = useCallback(() => { const newSortOrder = page.filter.sortOrder === SortOrder.ASC ? SortOrder.DESC : SortOrder.ASC; const updatedFilters = setFilter({ sortOrder: newSortOrder }); handleFilterChange(updatedFilters); }, [page.filter.sortOrder, handleFilterChange, setFilter]); const handleSetViewType = useCallback( (e: MouseEvent) => { if (!e.currentTarget?.value) return; const display = e.currentTarget.value as ListDisplayType; setPage({ list: { ...page, display: e.currentTarget.value as ListDisplayType } }); if (display === ListDisplayType.TABLE) { tableRef.current?.api.paginationSetPageSize(tableRef.current.props.infiniteInitialRowCount); setPagination({ currentPage: 0 }); } else if (display === ListDisplayType.TABLE_PAGINATED) { setPagination({ currentPage: 0 }); } }, [page, setPage, setPagination, tableRef], ); const handleSearch = debounce((e: ChangeEvent) => { const previousSearchTerm = page.filter.searchTerm; const searchTerm = e.target.value === '' ? undefined : e.target.value; const updatedFilters = setFilter({ searchTerm }); if (previousSearchTerm !== searchTerm) handleFilterChange(updatedFilters); }, 500); const handleTableColumns = (values: TableColumn[]) => { const existingColumns = page.table.columns; if (values.length === 0) { return setTable({ columns: [], }); } // If adding a column if (values.length > existingColumns.length) { const newColumn = { column: values[values.length - 1], width: 100 }; return setTable({ columns: [...existingColumns, newColumn] }); } // If removing a column const removed = existingColumns.filter((column) => !values.includes(column.column)); const newColumns = existingColumns.filter((column) => !removed.includes(column)); return setTable({ columns: newColumns }); }; const handleAutoFitColumns = (e: ChangeEvent) => { setTable({ autoFit: e.currentTarget.checked }); if (e.currentTarget.checked) { tableRef.current?.api.sizeColumnsToFit(); } }; const handleRowHeight = (e: number) => { setTable({ rowHeight: e }); }; return ( Display type Table Table (paginated) Item Size Table Columns column.column)} width={300} onChange={handleTableColumns} /> Auto Fit Columns {FILTERS[server?.type as keyof typeof FILTERS].map((filter) => ( {filter.name} ))} {server?.type === ServerType.JELLYFIN && ( {musicFoldersQuery.data?.map((folder) => ( {folder.name} ))} )} {server?.type === ServerType.NAVIDROME ? ( ) : ( )} Play Play last Play next Add to playlist ); };