import { 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 { useQueryClient } from '@tanstack/react-query'; import { ChangeEvent, MutableRefObject, useCallback, MouseEvent } from 'react'; import { RiArrowDownSLine, RiMoreFill, RiSortAsc, RiSortDesc } from 'react-icons/ri'; import { useParams } from 'react-router'; import styled from 'styled-components'; import { api } from '/@/renderer/api'; import { queryKeys } from '/@/renderer/api/query-keys'; import { PlaylistSongListQuery, SongListSort, SortOrder } from '/@/renderer/api/types'; import { Button, DropdownMenu, MultiSelect, PageHeader, Slider, SONG_TABLE_COLUMNS, Switch, Text, TextTitle, } from '/@/renderer/components'; import { usePlayQueueAdd } from '/@/renderer/features/player'; import { usePlaylistDetail } from '/@/renderer/features/playlists/queries/playlist-detail-query'; import { useContainerQuery } from '/@/renderer/hooks'; import { useCurrentServer, usePlaylistDetailStore, useSetPlaylistTablePagination, useSetPlaylistDetailTable, SongListFilter, useSetPlaylistDetailFilters, useSetPlaylistStore, } from '/@/renderer/store'; import { LibraryItem, ListDisplayType, Play, 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: 'Id', value: SongListSort.ID }, { 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 }, ]; const HeaderItems = styled.div` display: flex; flex-direction: row; justify-content: space-between; `; interface PlaylistDetailHeaderProps { tableRef: MutableRefObject; } export const PlaylistDetailSongListHeader = ({ tableRef }: PlaylistDetailHeaderProps) => { const { playlistId } = useParams() as { playlistId: string }; const queryClient = useQueryClient(); const server = useCurrentServer(); const setPage = useSetPlaylistStore(); const setFilter = useSetPlaylistDetailFilters(); const page = usePlaylistDetailStore(); const filters: Partial = { sortBy: page?.table.id[playlistId]?.filter?.sortBy || SongListSort.ID, sortOrder: page?.table.id[playlistId]?.filter?.sortOrder || SortOrder.ASC, }; const detailQuery = usePlaylistDetail({ id: playlistId }); const handlePlayQueueAdd = usePlayQueueAdd(); const cq = useContainerQuery(); const setPagination = useSetPlaylistTablePagination(); const setTable = useSetPlaylistDetailTable(); const sortByLabel = (server?.type && FILTERS[server.type as keyof typeof FILTERS].find((f) => f.value === filters.sortBy)?.name) || 'Unknown'; const sortOrderLabel = ORDER.find((o) => o.value === filters.sortOrder)?.name || 'Unknown'; const handleItemSize = (e: number) => { setTable({ rowHeight: e }); }; const handleFilterChange = useCallback( async (filters: SongListFilter) => { const dataSource: IDatasource = { getRows: async (params) => { const limit = params.endRow - params.startRow; const startIndex = params.startRow; const queryKey = queryKeys.playlists.songList(server?.id || '', playlistId, { id: playlistId, limit, startIndex, ...filters, }); const songsRes = await queryClient.fetchQuery(queryKey, async ({ signal }) => api.controller.getPlaylistSongList({ query: { id: playlistId, limit, startIndex, ...filters, }, server, signal, }), ); const songs = api.normalize.songList(songsRes, server); params.successCallback(songs?.items || [], songsRes?.totalRecordCount || undefined); }, rowCount: undefined, }; tableRef.current?.api.setDatasource(dataSource); tableRef.current?.api.purgeInfiniteCache(); tableRef.current?.api.ensureIndexVisible(0, 'top'); if (page.display === ListDisplayType.TABLE_PAGINATED) { setPagination({ currentPage: 0 }); } }, [tableRef, page.display, server, playlistId, queryClient, setPagination], ); 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(playlistId, { sortBy: e.currentTarget.value as SongListSort, sortOrder: sortOrder || SortOrder.ASC, }); handleFilterChange(updatedFilters); }, [handleFilterChange, playlistId, server?.type, setFilter], ); const handleToggleSortOrder = useCallback(() => { const newSortOrder = filters.sortOrder === SortOrder.ASC ? SortOrder.DESC : SortOrder.ASC; const updatedFilters = setFilter(playlistId, { sortOrder: newSortOrder }); handleFilterChange(updatedFilters); }, [filters.sortOrder, handleFilterChange, playlistId, setFilter]); const handleSetViewType = useCallback( (e: MouseEvent) => { if (!e.currentTarget?.value) return; setPage({ detail: { ...page, display: e.currentTarget.value as ListDisplayType } }); }, [page, setPage], ); 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 }; setTable({ columns: [...existingColumns, newColumn] }); } else { // If removing a column const removed = existingColumns.filter((column) => !values.includes(column.column)); const newColumns = existingColumns.filter((column) => !removed.includes(column)); setTable({ columns: newColumns }); } return tableRef.current?.api.sizeColumnsToFit(); }; const handleAutoFitColumns = (e: ChangeEvent) => { setTable({ autoFit: e.currentTarget.checked }); if (e.currentTarget.checked) { tableRef.current?.api.sizeColumnsToFit(); } }; const handlePlay = async (playType: Play) => { handlePlayQueueAdd?.({ byItemType: { id: [playlistId], type: LibraryItem.PLAYLIST }, play: playType, }); }; if (detailQuery.isLoading) return null; return ( Display type Table Table (paginated) Item size {(page.display === ListDisplayType.TABLE || page.display === ListDisplayType.TABLE_PAGINATED) && ( <> Table Columns column.column)} width={300} onChange={handleTableColumns} /> Auto Fit Columns )} {FILTERS[server?.type as keyof typeof FILTERS].map((filter) => ( {filter.name} ))} handlePlay(Play.NOW)}>Play handlePlay(Play.NEXT)} > Add to queue (last) handlePlay(Play.LAST)} > Add to queue (next) ); };