mirror of
https://github.com/jeffvli/feishin.git
synced 2026-05-09 20:29:36 +02:00
normalize controls onto lists
This commit is contained in:
@@ -75,6 +75,13 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.play-button.disabled,
|
||||||
|
.play-button.loading {
|
||||||
|
cursor: not-allowed;
|
||||||
|
opacity: 0.5;
|
||||||
|
transform: translate(-50%, -50%) scale(1);
|
||||||
|
}
|
||||||
|
|
||||||
.play-button.primary {
|
.play-button.primary {
|
||||||
left: 50%;
|
left: 50%;
|
||||||
width: 25%;
|
width: 25%;
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ import { MouseEvent } from 'react';
|
|||||||
|
|
||||||
import styles from './item-card-controls.module.css';
|
import styles from './item-card-controls.module.css';
|
||||||
|
|
||||||
|
import { ItemListStateActions } from '/@/renderer/components/item-list/helpers/item-list-state';
|
||||||
import { ItemControls } from '/@/renderer/components/item-list/types';
|
import { ItemControls } from '/@/renderer/components/item-list/types';
|
||||||
|
import { useIsPlayerFetching } from '/@/renderer/features/player/context/player-context';
|
||||||
import { animationVariants } from '/@/shared/components/animations/animation-variants';
|
import { animationVariants } from '/@/shared/components/animations/animation-variants';
|
||||||
import { AppIcon, Icon } from '/@/shared/components/icon/icon';
|
import { AppIcon, Icon } from '/@/shared/components/icon/icon';
|
||||||
import { Rating } from '/@/shared/components/rating/rating';
|
import { Rating } from '/@/shared/components/rating/rating';
|
||||||
@@ -19,7 +21,8 @@ import {
|
|||||||
import { Play } from '/@/shared/types/types';
|
import { Play } from '/@/shared/types/types';
|
||||||
|
|
||||||
interface ItemCardControlsProps {
|
interface ItemCardControlsProps {
|
||||||
controls: ItemControls;
|
controls?: ItemControls;
|
||||||
|
internalState?: ItemListStateActions;
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined;
|
item: Album | AlbumArtist | Artist | Playlist | Song | undefined;
|
||||||
itemType: LibraryItem;
|
itemType: LibraryItem;
|
||||||
type?: 'compact' | 'default' | 'poster';
|
type?: 'compact' | 'default' | 'poster';
|
||||||
@@ -48,58 +51,152 @@ const containerProps = {
|
|||||||
|
|
||||||
export const ItemCardControls = ({
|
export const ItemCardControls = ({
|
||||||
controls,
|
controls,
|
||||||
|
internalState,
|
||||||
item,
|
item,
|
||||||
itemType,
|
itemType,
|
||||||
type = 'default',
|
type = 'default',
|
||||||
}: ItemCardControlsProps) => {
|
}: ItemCardControlsProps) => {
|
||||||
|
const isPlayerFetching = useIsPlayerFetching();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<motion.div className={clsx(styles.container)} {...containerProps[type]}>
|
<motion.div className={clsx(styles.container)} {...containerProps[type]}>
|
||||||
<PlayButton
|
{controls?.onPlay && (
|
||||||
onClick={(e) => {
|
<>
|
||||||
e.stopPropagation();
|
<PlayButton
|
||||||
controls?.onPlay?.(item, itemType, Play.NOW, e);
|
disabled={isPlayerFetching}
|
||||||
}}
|
onClick={(e) => {
|
||||||
/>
|
e.stopPropagation();
|
||||||
<SecondaryPlayButton
|
|
||||||
className={styles.left}
|
if (!item) {
|
||||||
icon="mediaPlayNext"
|
return;
|
||||||
onClick={(e) => {
|
}
|
||||||
e.stopPropagation();
|
|
||||||
controls?.onPlay?.(item, itemType, Play.NEXT, e);
|
controls?.onPlay?.({
|
||||||
}}
|
event: e,
|
||||||
/>
|
internalState,
|
||||||
<SecondaryPlayButton
|
item,
|
||||||
className={styles.right}
|
itemType,
|
||||||
icon="mediaPlayLast"
|
playType: Play.NOW,
|
||||||
onClick={(e) => {
|
});
|
||||||
e.stopPropagation();
|
}}
|
||||||
controls?.onPlay?.(item, itemType, Play.LAST, e);
|
/>
|
||||||
}}
|
<SecondaryPlayButton
|
||||||
/>
|
className={styles.left}
|
||||||
<SecondaryButton
|
icon="mediaPlayNext"
|
||||||
className={styles.favorite}
|
onClick={(e) => {
|
||||||
icon="favorite"
|
e.stopPropagation();
|
||||||
onClick={(e) => {
|
|
||||||
e.stopPropagation();
|
if (!item) {
|
||||||
controls?.onFavorite?.(item, itemType, e);
|
return;
|
||||||
}}
|
}
|
||||||
/>
|
|
||||||
<Rating className={styles.rating} size="xs" />
|
controls?.onPlay?.({
|
||||||
<SecondaryButton
|
event: e,
|
||||||
className={styles.options}
|
internalState,
|
||||||
icon="ellipsisHorizontal"
|
item,
|
||||||
onClick={(e) => {
|
itemType,
|
||||||
e.stopPropagation();
|
playType: Play.NEXT,
|
||||||
controls?.onMore?.(item, itemType, e);
|
});
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{controls?.onItemExpand && (
|
<SecondaryPlayButton
|
||||||
|
className={styles.right}
|
||||||
|
icon="mediaPlayLast"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
if (!item) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
controls?.onPlay?.({
|
||||||
|
event: e,
|
||||||
|
internalState,
|
||||||
|
item,
|
||||||
|
itemType,
|
||||||
|
playType: Play.LAST,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{controls?.onFavorite && (
|
||||||
|
<SecondaryButton
|
||||||
|
className={styles.favorite}
|
||||||
|
icon="favorite"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
if (!item) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newFavorite = !(item as { userFavorite: boolean }).userFavorite;
|
||||||
|
controls?.onFavorite?.({
|
||||||
|
event: e,
|
||||||
|
favorite: newFavorite,
|
||||||
|
internalState,
|
||||||
|
item,
|
||||||
|
itemType,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{controls?.onRating && (
|
||||||
|
<Rating
|
||||||
|
className={styles.rating}
|
||||||
|
onChange={(rating) => {
|
||||||
|
if (!item) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let newRating = rating;
|
||||||
|
|
||||||
|
if (rating === (item as { userRating: number }).userRating) {
|
||||||
|
newRating = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
controls?.onRating?.({
|
||||||
|
event: null,
|
||||||
|
internalState,
|
||||||
|
item,
|
||||||
|
itemType,
|
||||||
|
rating: newRating,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
}}
|
||||||
|
size="xs"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{controls?.onMore && (
|
||||||
|
<SecondaryButton
|
||||||
|
className={styles.options}
|
||||||
|
icon="ellipsisHorizontal"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
controls?.onMore?.({
|
||||||
|
event: e,
|
||||||
|
internalState,
|
||||||
|
item,
|
||||||
|
itemType,
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{controls?.onExpand && (
|
||||||
<SecondaryButton
|
<SecondaryButton
|
||||||
className={styles.expand}
|
className={styles.expand}
|
||||||
icon="arrowDownS"
|
icon="arrowDownS"
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
controls?.onItemExpand?.(item, itemType, e);
|
controls?.onExpand?.({
|
||||||
|
event: e,
|
||||||
|
internalState,
|
||||||
|
item,
|
||||||
|
itemType,
|
||||||
|
});
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@@ -107,12 +204,26 @@ export const ItemCardControls = ({
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const PlayButton = ({ onClick }: { onClick?: (e: MouseEvent<HTMLButtonElement>) => void }) => {
|
const PlayButton = ({
|
||||||
|
disabled,
|
||||||
|
loading,
|
||||||
|
onClick,
|
||||||
|
}: {
|
||||||
|
disabled?: boolean;
|
||||||
|
loading?: boolean;
|
||||||
|
onClick?: (e: MouseEvent<HTMLButtonElement>) => void;
|
||||||
|
}) => {
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
className={clsx(styles.playButton, styles.primary)}
|
className={clsx(styles.playButton, styles.primary, {
|
||||||
|
[styles.disabled]: disabled,
|
||||||
|
})}
|
||||||
|
disabled={disabled}
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
|
if (disabled || loading) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
onClick?.(e);
|
onClick?.(e);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { generatePath, Link } from 'react-router';
|
|||||||
import styles from './item-card.module.css';
|
import styles from './item-card.module.css';
|
||||||
|
|
||||||
import { ItemCardControls } from '/@/renderer/components/item-card/item-card-controls';
|
import { ItemCardControls } from '/@/renderer/components/item-card/item-card-controls';
|
||||||
|
import { ItemListStateActions } from '/@/renderer/components/item-list/helpers/item-list-state';
|
||||||
import { ItemControls } from '/@/renderer/components/item-list/types';
|
import { ItemControls } from '/@/renderer/components/item-list/types';
|
||||||
import { AppRoute } from '/@/renderer/router/routes';
|
import { AppRoute } from '/@/renderer/router/routes';
|
||||||
import { Image } from '/@/shared/components/image/image';
|
import { Image } from '/@/shared/components/image/image';
|
||||||
@@ -21,24 +22,26 @@ import {
|
|||||||
Song,
|
Song,
|
||||||
} from '/@/shared/types/domain-types';
|
} from '/@/shared/types/domain-types';
|
||||||
|
|
||||||
type DataRow = {
|
export interface ItemCardProps {
|
||||||
format: (data: Album | AlbumArtist | Artist | Playlist | Song) => ReactNode | string;
|
controls?: ItemControls;
|
||||||
id: string;
|
|
||||||
isMuted?: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
interface ItemCardProps {
|
|
||||||
controls: ItemControls;
|
|
||||||
data: Album | AlbumArtist | Artist | Playlist | Song | undefined;
|
data: Album | AlbumArtist | Artist | Playlist | Song | undefined;
|
||||||
|
internalState?: ItemListStateActions;
|
||||||
isRound?: boolean;
|
isRound?: boolean;
|
||||||
itemType: LibraryItem;
|
itemType: LibraryItem;
|
||||||
type?: 'compact' | 'default' | 'poster';
|
type?: 'compact' | 'default' | 'poster';
|
||||||
withControls?: boolean;
|
withControls?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type DataRow = {
|
||||||
|
format: (data: Album | AlbumArtist | Artist | Playlist | Song) => ReactNode | string;
|
||||||
|
id: string;
|
||||||
|
isMuted?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
export const ItemCard = ({
|
export const ItemCard = ({
|
||||||
controls,
|
controls,
|
||||||
data,
|
data,
|
||||||
|
internalState,
|
||||||
isRound,
|
isRound,
|
||||||
itemType,
|
itemType,
|
||||||
type = 'poster',
|
type = 'poster',
|
||||||
@@ -54,6 +57,7 @@ export const ItemCard = ({
|
|||||||
controls={controls}
|
controls={controls}
|
||||||
data={data}
|
data={data}
|
||||||
imageUrl={imageUrl}
|
imageUrl={imageUrl}
|
||||||
|
internalState={internalState}
|
||||||
isRound={isRound}
|
isRound={isRound}
|
||||||
itemType={itemType}
|
itemType={itemType}
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -66,6 +70,7 @@ export const ItemCard = ({
|
|||||||
controls={controls}
|
controls={controls}
|
||||||
data={data}
|
data={data}
|
||||||
imageUrl={imageUrl}
|
imageUrl={imageUrl}
|
||||||
|
internalState={internalState}
|
||||||
isRound={isRound}
|
isRound={isRound}
|
||||||
itemType={itemType}
|
itemType={itemType}
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -79,6 +84,7 @@ export const ItemCard = ({
|
|||||||
controls={controls}
|
controls={controls}
|
||||||
data={data}
|
data={data}
|
||||||
imageUrl={imageUrl}
|
imageUrl={imageUrl}
|
||||||
|
internalState={internalState}
|
||||||
isRound={isRound}
|
isRound={isRound}
|
||||||
itemType={itemType}
|
itemType={itemType}
|
||||||
rows={rows}
|
rows={rows}
|
||||||
@@ -89,8 +95,9 @@ export const ItemCard = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
export interface ItemCardDerivativeProps extends Omit<ItemCardProps, 'type'> {
|
export interface ItemCardDerivativeProps extends Omit<ItemCardProps, 'type'> {
|
||||||
controls: ItemControls;
|
controls?: ItemControls;
|
||||||
imageUrl: string | undefined;
|
imageUrl: string | undefined;
|
||||||
|
internalState?: ItemListStateActions;
|
||||||
rows: DataRow[];
|
rows: DataRow[];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,7 +126,7 @@ const CompactItemCard = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
const handleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
controls?.onClick?.(data, itemType, e);
|
// controls?.onClick?.(data, itemType, e);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -195,7 +202,7 @@ const DefaultItemCard = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
const handleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
controls?.onClick?.(data, itemType, e);
|
// controls?.onClick?.(data, itemType, e);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -250,6 +257,7 @@ const PosterItemCard = ({
|
|||||||
controls,
|
controls,
|
||||||
data,
|
data,
|
||||||
imageUrl,
|
imageUrl,
|
||||||
|
internalState,
|
||||||
isRound,
|
isRound,
|
||||||
itemType,
|
itemType,
|
||||||
rows,
|
rows,
|
||||||
@@ -271,7 +279,7 @@ const PosterItemCard = ({
|
|||||||
};
|
};
|
||||||
|
|
||||||
const handleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
const handleClick = (e: React.MouseEvent<HTMLDivElement>) => {
|
||||||
controls?.onClick?.(data, itemType, e);
|
// controls?.onClick?.(data, itemType, e);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -289,6 +297,7 @@ const PosterItemCard = ({
|
|||||||
{withControls && showControls && data && (
|
{withControls && showControls && data && (
|
||||||
<ItemCardControls
|
<ItemCardControls
|
||||||
controls={controls}
|
controls={controls}
|
||||||
|
internalState={internalState}
|
||||||
item={data}
|
item={data}
|
||||||
itemType={itemType}
|
itemType={itemType}
|
||||||
type="poster"
|
type="poster"
|
||||||
|
|||||||
@@ -1,102 +1,104 @@
|
|||||||
import { MouseEvent } from 'react';
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
import {
|
import { ItemListItem } from '/@/renderer/components/item-list/helpers/item-list-state';
|
||||||
ItemListItem,
|
import { DefaultItemControlProps, ItemControls } from '/@/renderer/components/item-list/types';
|
||||||
ItemListStateActions,
|
import { usePlayerContext } from '/@/renderer/features/player/context/player-context';
|
||||||
} from '/@/renderer/components/item-list/helpers/item-list-state';
|
|
||||||
import { LibraryItem } from '/@/shared/types/domain-types';
|
|
||||||
import { Play } from '/@/shared/types/types';
|
import { Play } from '/@/shared/types/types';
|
||||||
|
|
||||||
const handleItemClick = (
|
export const useDefaultItemListControls = () => {
|
||||||
item: (ItemListItem & object) | undefined,
|
const player = usePlayerContext();
|
||||||
itemType: LibraryItem,
|
|
||||||
internalState: ItemListStateActions,
|
|
||||||
) => {
|
|
||||||
if (!item) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const itemListItem: ItemListItem = {
|
const controls: ItemControls = useMemo(() => {
|
||||||
id: item.id,
|
return {
|
||||||
itemType,
|
onClick: ({ internalState, item, itemType }: DefaultItemControlProps) => {
|
||||||
serverId: item.serverId,
|
if (!item) {
|
||||||
};
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Regular click - deselect all others and select only this item
|
const itemListItem: ItemListItem = {
|
||||||
// If this item is already the only selected item, deselect it
|
_serverId: item._serverId,
|
||||||
const selectedItems = internalState.getSelected();
|
id: item.id,
|
||||||
const isOnlySelected = selectedItems.length === 1 && selectedItems[0].id === item.id;
|
itemType,
|
||||||
|
};
|
||||||
|
|
||||||
if (isOnlySelected) {
|
// Regular click - deselect all others and select only this item
|
||||||
internalState.clearSelected();
|
// If this item is already the only selected item, deselect it
|
||||||
} else {
|
const selectedItems = internalState.getSelected();
|
||||||
internalState.setSelected([itemListItem]);
|
const isOnlySelected =
|
||||||
}
|
selectedItems.length === 1 && selectedItems[0].id === item.id;
|
||||||
};
|
|
||||||
|
if (isOnlySelected) {
|
||||||
const handleItemDoubleClick = (
|
internalState.clearSelected();
|
||||||
item: (ItemListItem & object) | undefined,
|
} else {
|
||||||
itemType: LibraryItem,
|
internalState.setSelected([itemListItem]);
|
||||||
internalState: ItemListStateActions,
|
}
|
||||||
) => {
|
},
|
||||||
console.log('handleItemDoubleClick', item, itemType, internalState);
|
|
||||||
};
|
onDoubleClick: ({ internalState, item, itemType }: DefaultItemControlProps) => {
|
||||||
|
console.log('onDoubleClick', item, itemType, internalState);
|
||||||
const handleItemExpand = (
|
},
|
||||||
item: (ItemListItem & object) | undefined,
|
|
||||||
itemType: LibraryItem,
|
onExpand: ({ internalState, item, itemType }: DefaultItemControlProps) => {
|
||||||
internalState: ItemListStateActions,
|
if (!item || !internalState) {
|
||||||
) => {
|
return;
|
||||||
if (!item) {
|
}
|
||||||
return;
|
|
||||||
}
|
return internalState?.toggleExpanded({
|
||||||
|
_serverId: item._serverId,
|
||||||
return internalState.toggleExpanded({
|
id: item.id,
|
||||||
id: item.id,
|
itemType,
|
||||||
itemType,
|
});
|
||||||
serverId: item.serverId,
|
},
|
||||||
});
|
|
||||||
};
|
onFavorite: ({
|
||||||
|
favorite,
|
||||||
const handleItemFavorite = (
|
item,
|
||||||
item: (ItemListItem & object) | undefined,
|
itemType,
|
||||||
itemType: LibraryItem,
|
}: DefaultItemControlProps & { favorite: boolean }) => {
|
||||||
internalState: ItemListStateActions,
|
if (!item) {
|
||||||
) => {
|
return;
|
||||||
console.log('handleItemFavorite', item, itemType, internalState);
|
}
|
||||||
};
|
|
||||||
|
player.setFavorite(item._serverId, [item.id], itemType, favorite);
|
||||||
const handleItemRating = (
|
},
|
||||||
item: (ItemListItem & object) | undefined,
|
|
||||||
itemType: LibraryItem,
|
onMore: ({ internalState, item, itemType }: DefaultItemControlProps) => {
|
||||||
internalState: ItemListStateActions,
|
console.log('handleItemMore', item, itemType, internalState);
|
||||||
) => {
|
},
|
||||||
console.log('handleItemRating', item, itemType, internalState);
|
|
||||||
};
|
onPlay: ({
|
||||||
|
item,
|
||||||
const handleItemMore = (
|
itemType,
|
||||||
item: (ItemListItem & object) | undefined,
|
playType,
|
||||||
itemType: LibraryItem,
|
}: DefaultItemControlProps & { playType: Play }) => {
|
||||||
internalState: ItemListStateActions,
|
if (!item) {
|
||||||
) => {
|
return;
|
||||||
console.log('handleItemMore', item, itemType, internalState);
|
}
|
||||||
};
|
|
||||||
|
player.addToQueueByFetch(item._serverId, [item.id], itemType, playType);
|
||||||
const handleItemPlay = (
|
},
|
||||||
item: (ItemListItem & object) | undefined,
|
|
||||||
itemType: LibraryItem,
|
onRating: ({
|
||||||
playType: Play,
|
item,
|
||||||
internalState: ItemListStateActions,
|
itemType,
|
||||||
) => {
|
rating,
|
||||||
console.log('handleItemPlay', item, itemType, playType, internalState);
|
}: DefaultItemControlProps & { rating: number }) => {
|
||||||
};
|
if (!item) {
|
||||||
|
return;
|
||||||
export const itemListControls = {
|
}
|
||||||
handleItemClick,
|
|
||||||
handleItemDoubleClick,
|
const previousRating = (item as { userRating: number }).userRating || 0;
|
||||||
handleItemExpand,
|
|
||||||
handleItemFavorite,
|
let newRating = rating;
|
||||||
handleItemMore,
|
|
||||||
handleItemPlay,
|
if (previousRating === rating) {
|
||||||
handleItemRating,
|
newRating = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
player.setRating(item._serverId, [item.id], itemType, newRating);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}, [player]);
|
||||||
|
|
||||||
|
return controls;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -13,9 +13,9 @@ export type ItemListAction =
|
|||||||
| { type: 'CLEAR_SELECTED' };
|
| { type: 'CLEAR_SELECTED' };
|
||||||
|
|
||||||
export interface ItemListItem {
|
export interface ItemListItem {
|
||||||
|
_serverId: string;
|
||||||
id: string;
|
id: string;
|
||||||
itemType: LibraryItem;
|
itemType: LibraryItem;
|
||||||
serverId: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ItemListState {
|
export interface ItemListState {
|
||||||
@@ -81,6 +81,8 @@ export const itemListReducer = (state: ItemListState, action: ItemListAction): I
|
|||||||
const newExpanded = new Set<string>();
|
const newExpanded = new Set<string>();
|
||||||
const newExpandedItems = new Map<string, ItemListItem>();
|
const newExpandedItems = new Map<string, ItemListItem>();
|
||||||
|
|
||||||
|
console.log('SET_EXPANDED', action.payload);
|
||||||
|
|
||||||
if (action.payload.length > 0) {
|
if (action.payload.length > 0) {
|
||||||
const firstItem = action.payload[0];
|
const firstItem = action.payload[0];
|
||||||
newExpanded.add(firstItem.id);
|
newExpanded.add(firstItem.id);
|
||||||
@@ -158,9 +160,6 @@ export const itemListReducer = (state: ItemListState, action: ItemListAction): I
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Initial state for item grid
|
|
||||||
*/
|
|
||||||
export const initialItemListState: ItemListState = {
|
export const initialItemListState: ItemListState = {
|
||||||
expanded: new Set(),
|
expanded: new Set(),
|
||||||
expandedItems: new Map(),
|
expandedItems: new Map(),
|
||||||
|
|||||||
@@ -125,7 +125,7 @@ export const ItemDetailList = ({
|
|||||||
internalState.toggleExpanded({
|
internalState.toggleExpanded({
|
||||||
id: item.id as string,
|
id: item.id as string,
|
||||||
itemType: itemType,
|
itemType: itemType,
|
||||||
serverId: item.serverId as string,
|
_serverId: item.serverId as string,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -24,12 +24,16 @@ import {
|
|||||||
ListOnScrollProps,
|
ListOnScrollProps,
|
||||||
} from 'react-window';
|
} from 'react-window';
|
||||||
|
|
||||||
import { ExpandedListContainer } from '../expanded-list-container';
|
|
||||||
import styles from './item-grid-list.module.css';
|
import styles from './item-grid-list.module.css';
|
||||||
|
|
||||||
import { getDataRowsCount, ItemCard } from '/@/renderer/components/item-card/item-card';
|
import {
|
||||||
|
getDataRowsCount,
|
||||||
|
ItemCard,
|
||||||
|
ItemCardProps,
|
||||||
|
} from '/@/renderer/components/item-card/item-card';
|
||||||
|
import { ExpandedListContainer } from '/@/renderer/components/item-list/expanded-list-container';
|
||||||
import { ExpandedListItem } from '/@/renderer/components/item-list/expanded-list-item';
|
import { ExpandedListItem } from '/@/renderer/components/item-list/expanded-list-item';
|
||||||
import { itemListControls } from '/@/renderer/components/item-list/helpers/item-list-controls';
|
import { useDefaultItemListControls } from '/@/renderer/components/item-list/helpers/item-list-controls';
|
||||||
import {
|
import {
|
||||||
ItemListStateActions,
|
ItemListStateActions,
|
||||||
useItemListState,
|
useItemListState,
|
||||||
@@ -38,6 +42,7 @@ import { ItemControls, ItemListHandle } from '/@/renderer/components/item-list/t
|
|||||||
import { LibraryItem } from '/@/shared/types/domain-types';
|
import { LibraryItem } from '/@/shared/types/domain-types';
|
||||||
|
|
||||||
interface VirtualizedGridListProps {
|
interface VirtualizedGridListProps {
|
||||||
|
controls: ItemControls;
|
||||||
data: unknown[];
|
data: unknown[];
|
||||||
enableExpansion: boolean;
|
enableExpansion: boolean;
|
||||||
enableSelection: boolean;
|
enableSelection: boolean;
|
||||||
@@ -59,6 +64,7 @@ interface VirtualizedGridListProps {
|
|||||||
|
|
||||||
const VirtualizedGridList = React.memo(
|
const VirtualizedGridList = React.memo(
|
||||||
({
|
({
|
||||||
|
controls,
|
||||||
data,
|
data,
|
||||||
enableExpansion,
|
enableExpansion,
|
||||||
enableSelection,
|
enableSelection,
|
||||||
@@ -76,50 +82,7 @@ const VirtualizedGridList = React.memo(
|
|||||||
const itemData: GridItemProps = useMemo(() => {
|
const itemData: GridItemProps = useMemo(() => {
|
||||||
return {
|
return {
|
||||||
columns: tableMeta?.columnCount || 0,
|
columns: tableMeta?.columnCount || 0,
|
||||||
controls: {
|
controls,
|
||||||
onClick: enableSelection
|
|
||||||
? (item, itemType) => {
|
|
||||||
return itemListControls.handleItemClick(
|
|
||||||
item,
|
|
||||||
itemType,
|
|
||||||
internalState,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
: undefined,
|
|
||||||
onDoubleClick: (item, itemType) => {
|
|
||||||
return itemListControls.handleItemDoubleClick(
|
|
||||||
item,
|
|
||||||
itemType,
|
|
||||||
internalState,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
onFavorite: (item, itemType) => {
|
|
||||||
return itemListControls.handleItemFavorite(item, itemType, internalState);
|
|
||||||
},
|
|
||||||
onItemExpand: enableExpansion
|
|
||||||
? (item, itemType) => {
|
|
||||||
return itemListControls.handleItemExpand(
|
|
||||||
item,
|
|
||||||
itemType,
|
|
||||||
internalState,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
: undefined,
|
|
||||||
onMore: (item, itemType) => {
|
|
||||||
return itemListControls.handleItemMore(item, itemType, internalState);
|
|
||||||
},
|
|
||||||
onPlay: (item, itemType, playType) => {
|
|
||||||
return itemListControls.handleItemPlay(
|
|
||||||
item,
|
|
||||||
itemType,
|
|
||||||
playType,
|
|
||||||
internalState,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
onRating: (item, itemType) => {
|
|
||||||
return itemListControls.handleItemRating(item, itemType, internalState);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
data,
|
data,
|
||||||
enableExpansion,
|
enableExpansion,
|
||||||
enableSelection,
|
enableSelection,
|
||||||
@@ -128,7 +91,16 @@ const VirtualizedGridList = React.memo(
|
|||||||
itemType,
|
itemType,
|
||||||
tableMeta,
|
tableMeta,
|
||||||
};
|
};
|
||||||
}, [enableSelection, enableExpansion, internalState, tableMeta, data, itemType, gap]);
|
}, [
|
||||||
|
tableMeta,
|
||||||
|
controls,
|
||||||
|
data,
|
||||||
|
enableExpansion,
|
||||||
|
enableSelection,
|
||||||
|
gap,
|
||||||
|
internalState,
|
||||||
|
itemType,
|
||||||
|
]);
|
||||||
|
|
||||||
const debouncedOnScrollEnd = useMemo(
|
const debouncedOnScrollEnd = useMemo(
|
||||||
() =>
|
() =>
|
||||||
@@ -253,7 +225,7 @@ const createThrottledSetTableMeta = (itemsPerRow?: number) => {
|
|||||||
|
|
||||||
export interface GridItemProps {
|
export interface GridItemProps {
|
||||||
columns: number;
|
columns: number;
|
||||||
controls: ItemControls;
|
controls: ItemCardProps['controls'];
|
||||||
data: any[];
|
data: any[];
|
||||||
enableExpansion?: boolean;
|
enableExpansion?: boolean;
|
||||||
enableSelection?: boolean;
|
enableSelection?: boolean;
|
||||||
@@ -354,6 +326,8 @@ export const ItemGridList = ({
|
|||||||
throttledSetTableMeta(containerWidth, data.length, itemType, setTableMeta);
|
throttledSetTableMeta(containerWidth, data.length, itemType, setTableMeta);
|
||||||
}, [containerWidth, data.length, itemType, throttledSetTableMeta]);
|
}, [containerWidth, data.length, itemType, throttledSetTableMeta]);
|
||||||
|
|
||||||
|
const controls = useDefaultItemListControls();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={styles.itemGridContainer}
|
className={styles.itemGridContainer}
|
||||||
@@ -361,6 +335,7 @@ export const ItemGridList = ({
|
|||||||
ref={mergedContainerRef}
|
ref={mergedContainerRef}
|
||||||
>
|
>
|
||||||
<VirtualizedGridList
|
<VirtualizedGridList
|
||||||
|
controls={controls}
|
||||||
data={data}
|
data={data}
|
||||||
enableExpansion={enableExpansion}
|
enableExpansion={enableExpansion}
|
||||||
enableSelection={enableSelection}
|
enableSelection={enableSelection}
|
||||||
@@ -411,7 +386,13 @@ const ListComponent = memo((props: ListChildComponentProps<GridItemProps>) => {
|
|||||||
key={`card-${i}-${index}`}
|
key={`card-${i}-${index}`}
|
||||||
style={{ '--columns': columns } as CSSProperties}
|
style={{ '--columns': columns } as CSSProperties}
|
||||||
>
|
>
|
||||||
<ItemCard controls={controls} data={data[i]} itemType={itemType} withControls />
|
<ItemCard
|
||||||
|
controls={controls}
|
||||||
|
data={data[i]}
|
||||||
|
internalState={props.data.internalState}
|
||||||
|
itemType={itemType}
|
||||||
|
withControls
|
||||||
|
/>
|
||||||
</div>,
|
</div>,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
|
|||||||
@@ -2,19 +2,14 @@ import {
|
|||||||
ItemTableListInnerColumn,
|
ItemTableListInnerColumn,
|
||||||
TableColumnContainer,
|
TableColumnContainer,
|
||||||
} from '/@/renderer/components/item-list/item-table-list/item-table-list-column';
|
} from '/@/renderer/components/item-list/item-table-list/item-table-list-column';
|
||||||
import { useCreateFavorite } from '/@/renderer/features/shared/mutations/create-favorite-mutation';
|
import { ItemListItem } from '/@/renderer/components/item-list/types';
|
||||||
import { useDeleteFavorite } from '/@/renderer/features/shared/mutations/delete-favorite-mutation';
|
|
||||||
import { ActionIcon } from '/@/shared/components/action-icon/action-icon';
|
import { ActionIcon } from '/@/shared/components/action-icon/action-icon';
|
||||||
import { LibraryItem } from '/@/shared/types/domain-types';
|
|
||||||
|
|
||||||
export const FavoriteColumn = (props: ItemTableListInnerColumn) => {
|
export const FavoriteColumn = (props: ItemTableListInnerColumn) => {
|
||||||
const row: boolean | undefined = (props.data as (any | undefined)[])[props.rowIndex]?.[
|
const row: boolean | undefined = (props.data as (any | undefined)[])[props.rowIndex]?.[
|
||||||
props.columns[props.columnIndex].id
|
props.columns[props.columnIndex].id
|
||||||
];
|
];
|
||||||
|
|
||||||
const createFavorite = useCreateFavorite({});
|
|
||||||
const deleteFavorite = useDeleteFavorite({});
|
|
||||||
|
|
||||||
if (typeof row === 'boolean') {
|
if (typeof row === 'boolean') {
|
||||||
return (
|
return (
|
||||||
<TableColumnContainer {...props}>
|
<TableColumnContainer {...props}>
|
||||||
@@ -26,36 +21,14 @@ export const FavoriteColumn = (props: ItemTableListInnerColumn) => {
|
|||||||
fill: row ? 'primary' : undefined,
|
fill: row ? 'primary' : undefined,
|
||||||
size: 'md',
|
size: 'md',
|
||||||
}}
|
}}
|
||||||
onClick={() => {
|
onClick={(event) => {
|
||||||
if (!props.data?.[props.rowIndex]) {
|
props.controls.onFavorite?.({
|
||||||
return;
|
event,
|
||||||
}
|
favorite: !row,
|
||||||
|
internalState: props.internalState,
|
||||||
if (row) {
|
item: props.data[props.rowIndex] as ItemListItem,
|
||||||
deleteFavorite.mutate({
|
itemType: props.itemType,
|
||||||
apiClientProps: {
|
});
|
||||||
serverId: (props.data as any)[props.rowIndex]
|
|
||||||
.serverId as string,
|
|
||||||
},
|
|
||||||
query: {
|
|
||||||
id: [(props.data as any)[props.rowIndex].id as string],
|
|
||||||
type: (props.data as any)[props.rowIndex]
|
|
||||||
.itemType as LibraryItem,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
createFavorite.mutate({
|
|
||||||
apiClientProps: {
|
|
||||||
serverId: (props.data as any)[props.rowIndex]
|
|
||||||
.serverId as string,
|
|
||||||
},
|
|
||||||
query: {
|
|
||||||
id: [(props.data as any)[props.rowIndex].id as string],
|
|
||||||
type: (props.data as any)[props.rowIndex]
|
|
||||||
.itemType as LibraryItem,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}}
|
}}
|
||||||
size="xs"
|
size="xs"
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ import {
|
|||||||
ItemTableListInnerColumn,
|
ItemTableListInnerColumn,
|
||||||
TableColumnContainer,
|
TableColumnContainer,
|
||||||
} from '/@/renderer/components/item-list/item-table-list/item-table-list-column';
|
} from '/@/renderer/components/item-list/item-table-list/item-table-list-column';
|
||||||
import { useSetRating } from '/@/renderer/features/shared/mutations/set-rating-mutation';
|
import { ItemListItem } from '/@/renderer/components/item-list/types';
|
||||||
import { Rating } from '/@/shared/components/rating/rating';
|
import { Rating } from '/@/shared/components/rating/rating';
|
||||||
|
|
||||||
export const RatingColumn = (props: ItemTableListInnerColumn) => {
|
export const RatingColumn = (props: ItemTableListInnerColumn) => {
|
||||||
@@ -10,35 +10,20 @@ export const RatingColumn = (props: ItemTableListInnerColumn) => {
|
|||||||
props.columns[props.columnIndex].id
|
props.columns[props.columnIndex].id
|
||||||
];
|
];
|
||||||
|
|
||||||
const setRatingMutation = useSetRating({});
|
|
||||||
|
|
||||||
const handleChangeRating = (rating: number) => {
|
|
||||||
const previousRating = row || 0;
|
|
||||||
|
|
||||||
let newRating = rating;
|
|
||||||
|
|
||||||
if (previousRating === rating) {
|
|
||||||
newRating = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const item = props.data[props.rowIndex] as any;
|
|
||||||
|
|
||||||
setRatingMutation.mutate({
|
|
||||||
apiClientProps: { serverId: item.serverId as string },
|
|
||||||
query: {
|
|
||||||
id: [item.id],
|
|
||||||
rating: newRating,
|
|
||||||
type: item.itemType,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
if (typeof row === 'number' || row === null) {
|
if (typeof row === 'number' || row === null) {
|
||||||
return (
|
return (
|
||||||
<TableColumnContainer {...props}>
|
<TableColumnContainer {...props}>
|
||||||
<Rating
|
<Rating
|
||||||
className={row ? undefined : 'hover-only-flex'}
|
className={row ? undefined : 'hover-only-flex'}
|
||||||
onChange={handleChangeRating}
|
onChange={(rating) => {
|
||||||
|
props.controls.onRating?.({
|
||||||
|
event: null,
|
||||||
|
internalState: props.internalState,
|
||||||
|
item: props.data[props.rowIndex] as ItemListItem,
|
||||||
|
itemType: props.itemType,
|
||||||
|
rating,
|
||||||
|
});
|
||||||
|
}}
|
||||||
size="xs"
|
size="xs"
|
||||||
value={row || 0}
|
value={row || 0}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -21,11 +21,7 @@ export const RowIndexColumn = (props: ItemTableListInnerColumn) => {
|
|||||||
icon="arrowDownS"
|
icon="arrowDownS"
|
||||||
iconProps={{ color: 'muted', size: 'md' }}
|
iconProps={{ color: 'muted', size: 'md' }}
|
||||||
onClick={(e) =>
|
onClick={(e) =>
|
||||||
controls.onItemExpand?.(
|
controls.onExpand?.(props.data[props.rowIndex] as any, props.itemType, e)
|
||||||
props.data[props.rowIndex] as any,
|
|
||||||
props.itemType,
|
|
||||||
e,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
size="xs"
|
size="xs"
|
||||||
variant="subtle"
|
variant="subtle"
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import { CellComponentProps } from 'react-window-v2';
|
|||||||
import styles from './item-table-list-column.module.css';
|
import styles from './item-table-list-column.module.css';
|
||||||
|
|
||||||
import i18n from '/@/i18n/i18n';
|
import i18n from '/@/i18n/i18n';
|
||||||
import { itemListControls } from '/@/renderer/components/item-list/helpers/item-list-controls';
|
|
||||||
import { ActionsColumn } from '/@/renderer/components/item-list/item-table-list/columns/actions-column';
|
import { ActionsColumn } from '/@/renderer/components/item-list/item-table-list/columns/actions-column';
|
||||||
import { AlbumArtistsColumn } from '/@/renderer/components/item-list/item-table-list/columns/album-artists-column';
|
import { AlbumArtistsColumn } from '/@/renderer/components/item-list/item-table-list/columns/album-artists-column';
|
||||||
import { ArtistsColumn } from '/@/renderer/components/item-list/item-table-list/columns/artists-column';
|
import { ArtistsColumn } from '/@/renderer/components/item-list/item-table-list/columns/artists-column';
|
||||||
@@ -29,7 +28,7 @@ import { TextColumn } from '/@/renderer/components/item-list/item-table-list/col
|
|||||||
import { TitleColumn } from '/@/renderer/components/item-list/item-table-list/columns/title-column';
|
import { TitleColumn } from '/@/renderer/components/item-list/item-table-list/columns/title-column';
|
||||||
import { TitleCombinedColumn } from '/@/renderer/components/item-list/item-table-list/columns/title-combined-column';
|
import { TitleCombinedColumn } from '/@/renderer/components/item-list/item-table-list/columns/title-combined-column';
|
||||||
import { TableItemProps } from '/@/renderer/components/item-list/item-table-list/item-table-list';
|
import { TableItemProps } from '/@/renderer/components/item-list/item-table-list/item-table-list';
|
||||||
import { ItemControls } from '/@/renderer/components/item-list/types';
|
import { ItemControls, ItemListItem } from '/@/renderer/components/item-list/types';
|
||||||
import { Icon } from '/@/shared/components/icon/icon';
|
import { Icon } from '/@/shared/components/icon/icon';
|
||||||
import { Skeleton } from '/@/shared/components/skeleton/skeleton';
|
import { Skeleton } from '/@/shared/components/skeleton/skeleton';
|
||||||
import { Text } from '/@/shared/components/text/text';
|
import { Text } from '/@/shared/components/text/text';
|
||||||
@@ -47,36 +46,10 @@ export const ItemTableListColumn = (props: ItemTableListColumn) => {
|
|||||||
|
|
||||||
const isHeaderEnabled = !!props.enableHeader;
|
const isHeaderEnabled = !!props.enableHeader;
|
||||||
|
|
||||||
const controls: ItemControls = {
|
const controls = props.controls;
|
||||||
onClick: (item, itemType, event) => {
|
|
||||||
if (props.onRowClick && item) {
|
|
||||||
props.onRowClick(item, event);
|
|
||||||
} else {
|
|
||||||
itemListControls.handleItemClick(item, itemType, props.internalState);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
onDoubleClick: (item, itemType) =>
|
|
||||||
itemListControls.handleItemDoubleClick(item, itemType, props.internalState),
|
|
||||||
onFavorite: (item, itemType) =>
|
|
||||||
itemListControls.handleItemFavorite(item, itemType, props.internalState),
|
|
||||||
onItemExpand: (item, itemType) =>
|
|
||||||
itemListControls.handleItemExpand(item, itemType, props.internalState),
|
|
||||||
onMore: (item, itemType) =>
|
|
||||||
itemListControls.handleItemMore(item, itemType, props.internalState),
|
|
||||||
onPlay: (item, itemType, playType) =>
|
|
||||||
itemListControls.handleItemPlay(item, itemType, playType, props.internalState),
|
|
||||||
onRating: (item, itemType) =>
|
|
||||||
itemListControls.handleItemRating(item, itemType, props.internalState),
|
|
||||||
};
|
|
||||||
|
|
||||||
if (isHeaderEnabled && props.rowIndex === 0) {
|
if (isHeaderEnabled && props.rowIndex === 0) {
|
||||||
return (
|
return <TableColumnHeaderContainer {...props} controls={controls} type={type} />;
|
||||||
<TableColumnHeaderContainer
|
|
||||||
{...props}
|
|
||||||
controls={controls}
|
|
||||||
type={type}
|
|
||||||
></TableColumnHeaderContainer>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
@@ -211,7 +184,12 @@ export const TableColumnTextContainer = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (isDataRow && item && props.enableSelection) {
|
if (isDataRow && item && props.enableSelection) {
|
||||||
props.controls.onClick?.(item as any, props.itemType, event);
|
props.controls.onClick?.({
|
||||||
|
event,
|
||||||
|
internalState: props.internalState,
|
||||||
|
item: item as ItemListItem,
|
||||||
|
itemType: props.itemType,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -315,7 +293,12 @@ export const TableColumnContainer = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (isDataRow && item && props.enableSelection) {
|
if (isDataRow && item && props.enableSelection) {
|
||||||
props.controls.onClick?.(item as any, props.itemType, event);
|
props.controls.onClick?.({
|
||||||
|
event,
|
||||||
|
internalState: props.internalState,
|
||||||
|
item: item as ItemListItem,
|
||||||
|
itemType: props.itemType,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -21,19 +21,25 @@ import styles from './item-table-list.module.css';
|
|||||||
|
|
||||||
import { ExpandedListContainer } from '/@/renderer/components/item-list/expanded-list-container';
|
import { ExpandedListContainer } from '/@/renderer/components/item-list/expanded-list-container';
|
||||||
import { ExpandedListItem } from '/@/renderer/components/item-list/expanded-list-item';
|
import { ExpandedListItem } from '/@/renderer/components/item-list/expanded-list-item';
|
||||||
|
import { useDefaultItemListControls } from '/@/renderer/components/item-list/helpers/item-list-controls';
|
||||||
import {
|
import {
|
||||||
ItemListItem,
|
ItemListItem,
|
||||||
ItemListStateActions,
|
ItemListStateActions,
|
||||||
useItemListState,
|
useItemListState,
|
||||||
} from '/@/renderer/components/item-list/helpers/item-list-state';
|
} from '/@/renderer/components/item-list/helpers/item-list-state';
|
||||||
import { parseTableColumns } from '/@/renderer/components/item-list/helpers/parse-table-columns';
|
import { parseTableColumns } from '/@/renderer/components/item-list/helpers/parse-table-columns';
|
||||||
import { ItemListHandle, ItemTableListColumnConfig } from '/@/renderer/components/item-list/types';
|
import {
|
||||||
|
ItemControls,
|
||||||
|
ItemListHandle,
|
||||||
|
ItemTableListColumnConfig,
|
||||||
|
} from '/@/renderer/components/item-list/types';
|
||||||
import { LibraryItem } from '/@/shared/types/domain-types';
|
import { LibraryItem } from '/@/shared/types/domain-types';
|
||||||
|
|
||||||
interface VirtualizedTableGridProps {
|
interface VirtualizedTableGridProps {
|
||||||
calculatedColumnWidths: number[];
|
calculatedColumnWidths: number[];
|
||||||
CellComponent: JSXElementConstructor<CellComponentProps<TableItemProps>>;
|
CellComponent: JSXElementConstructor<CellComponentProps<TableItemProps>>;
|
||||||
cellPadding: 'lg' | 'md' | 'sm' | 'xl' | 'xs';
|
cellPadding: 'lg' | 'md' | 'sm' | 'xl' | 'xs';
|
||||||
|
controls: ItemControls;
|
||||||
data: unknown[];
|
data: unknown[];
|
||||||
enableAlternateRowColors: boolean;
|
enableAlternateRowColors: boolean;
|
||||||
enableExpansion: boolean;
|
enableExpansion: boolean;
|
||||||
@@ -68,6 +74,7 @@ const VirtualizedTableGrid = React.memo(
|
|||||||
calculatedColumnWidths,
|
calculatedColumnWidths,
|
||||||
CellComponent,
|
CellComponent,
|
||||||
cellPadding,
|
cellPadding,
|
||||||
|
controls,
|
||||||
data,
|
data,
|
||||||
enableAlternateRowColors,
|
enableAlternateRowColors,
|
||||||
enableExpansion,
|
enableExpansion,
|
||||||
@@ -105,6 +112,7 @@ const VirtualizedTableGrid = React.memo(
|
|||||||
() => ({
|
() => ({
|
||||||
cellPadding,
|
cellPadding,
|
||||||
columns: parsedColumns,
|
columns: parsedColumns,
|
||||||
|
controls,
|
||||||
data: enableHeader ? [null, ...data] : data,
|
data: enableHeader ? [null, ...data] : data,
|
||||||
enableAlternateRowColors,
|
enableAlternateRowColors,
|
||||||
enableExpansion,
|
enableExpansion,
|
||||||
@@ -121,6 +129,7 @@ const VirtualizedTableGrid = React.memo(
|
|||||||
}),
|
}),
|
||||||
[
|
[
|
||||||
cellPadding,
|
cellPadding,
|
||||||
|
controls,
|
||||||
parsedColumns,
|
parsedColumns,
|
||||||
enableHeader,
|
enableHeader,
|
||||||
data,
|
data,
|
||||||
@@ -402,6 +411,7 @@ VirtualizedTableGrid.displayName = 'VirtualizedTableGrid';
|
|||||||
export interface TableItemProps {
|
export interface TableItemProps {
|
||||||
cellPadding?: ItemTableListProps['cellPadding'];
|
cellPadding?: ItemTableListProps['cellPadding'];
|
||||||
columns: ItemTableListColumnConfig[];
|
columns: ItemTableListColumnConfig[];
|
||||||
|
controls: ItemControls;
|
||||||
data: ItemTableListProps['data'];
|
data: ItemTableListProps['data'];
|
||||||
enableAlternateRowColors?: ItemTableListProps['enableAlternateRowColors'];
|
enableAlternateRowColors?: ItemTableListProps['enableAlternateRowColors'];
|
||||||
enableExpansion?: ItemTableListProps['enableExpansion'];
|
enableExpansion?: ItemTableListProps['enableExpansion'];
|
||||||
@@ -918,9 +928,9 @@ export const ItemTableList = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const itemListItem: ItemListItem = {
|
const itemListItem: ItemListItem = {
|
||||||
|
_serverId: item.serverId,
|
||||||
id: item.id,
|
id: item.id,
|
||||||
itemType,
|
itemType,
|
||||||
serverId: item.serverId,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Check if ctrl/cmd key is held for multi-selection
|
// Check if ctrl/cmd key is held for multi-selection
|
||||||
@@ -971,9 +981,9 @@ export const ItemTableList = ({
|
|||||||
'serverId' in rangeItem
|
'serverId' in rangeItem
|
||||||
) {
|
) {
|
||||||
rangeItems.push({
|
rangeItems.push({
|
||||||
|
_serverId: (rangeItem as any).serverId,
|
||||||
id: (rangeItem as any).id,
|
id: (rangeItem as any).id,
|
||||||
itemType,
|
itemType,
|
||||||
serverId: (rangeItem as any).serverId,
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1071,12 +1081,15 @@ export const ItemTableList = ({
|
|||||||
handleRef.current = imperativeHandle;
|
handleRef.current = imperativeHandle;
|
||||||
}, [imperativeHandle]);
|
}, [imperativeHandle]);
|
||||||
|
|
||||||
|
const controls = useDefaultItemListControls();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={styles.itemTableListContainer}>
|
<div className={styles.itemTableListContainer}>
|
||||||
<VirtualizedTableGrid
|
<VirtualizedTableGrid
|
||||||
calculatedColumnWidths={calculatedColumnWidths}
|
calculatedColumnWidths={calculatedColumnWidths}
|
||||||
CellComponent={CellComponent}
|
CellComponent={CellComponent}
|
||||||
cellPadding={cellPadding}
|
cellPadding={cellPadding}
|
||||||
|
controls={controls}
|
||||||
data={data}
|
data={data}
|
||||||
enableAlternateRowColors={enableAlternateRowColors}
|
enableAlternateRowColors={enableAlternateRowColors}
|
||||||
enableExpansion={enableExpansion}
|
enableExpansion={enableExpansion}
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { MouseEvent } from 'react';
|
|
||||||
|
|
||||||
import { ItemListStateActions } from '/@/renderer/components/item-list/helpers/item-list-state';
|
import { ItemListStateActions } from '/@/renderer/components/item-list/helpers/item-list-state';
|
||||||
import {
|
import {
|
||||||
Album,
|
Album,
|
||||||
@@ -11,43 +9,35 @@ import {
|
|||||||
} from '/@/shared/types/domain-types';
|
} from '/@/shared/types/domain-types';
|
||||||
import { Play, TableColumn } from '/@/shared/types/types';
|
import { Play, TableColumn } from '/@/shared/types/types';
|
||||||
|
|
||||||
|
export interface DefaultItemControlProps {
|
||||||
|
event: null | React.MouseEvent<unknown>;
|
||||||
|
internalState?: ItemListStateActions;
|
||||||
|
item: ItemListItem | undefined;
|
||||||
|
itemType: LibraryItem;
|
||||||
|
}
|
||||||
|
|
||||||
export interface ItemControls {
|
export interface ItemControls {
|
||||||
onClick?: (
|
onClick?: ({ internalState, item, itemType }: DefaultItemControlProps) => void;
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
onDoubleClick?: ({ internalState, item, itemType }: DefaultItemControlProps) => void;
|
||||||
itemType: LibraryItem,
|
onExpand?: ({ internalState, item, itemType }: DefaultItemControlProps) => void;
|
||||||
e: MouseEvent<HTMLDivElement>,
|
onFavorite?: ({
|
||||||
) => void;
|
internalState,
|
||||||
onDoubleClick?: (
|
item,
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
itemType,
|
||||||
itemType: LibraryItem,
|
}: DefaultItemControlProps & { favorite: boolean }) => void;
|
||||||
e: MouseEvent<HTMLDivElement>,
|
onMore?: ({ internalState, item, itemType }: DefaultItemControlProps) => void;
|
||||||
) => void;
|
onPlay?: ({
|
||||||
onFavorite?: (
|
internalState,
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
item,
|
||||||
itemType: LibraryItem,
|
itemType,
|
||||||
e: MouseEvent<HTMLButtonElement>,
|
playType,
|
||||||
) => void;
|
}: DefaultItemControlProps & { playType: Play }) => void;
|
||||||
onItemExpand?: (
|
onRating?: ({
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
internalState,
|
||||||
itemType: LibraryItem,
|
item,
|
||||||
e: MouseEvent<HTMLButtonElement>,
|
itemType,
|
||||||
) => void;
|
rating,
|
||||||
onMore?: (
|
}: DefaultItemControlProps & { rating: number }) => void;
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
|
||||||
itemType: LibraryItem,
|
|
||||||
e: MouseEvent<HTMLButtonElement>,
|
|
||||||
) => void;
|
|
||||||
onPlay?: (
|
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
|
||||||
itemType: LibraryItem,
|
|
||||||
playType: Play,
|
|
||||||
e: MouseEvent<HTMLButtonElement>,
|
|
||||||
) => void;
|
|
||||||
onRating?: (
|
|
||||||
item: Album | AlbumArtist | Artist | Playlist | Song | undefined,
|
|
||||||
itemType: LibraryItem,
|
|
||||||
e: MouseEvent<HTMLDivElement>,
|
|
||||||
) => void;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ItemListComponentProps<TQuery> {
|
export interface ItemListComponentProps<TQuery> {
|
||||||
@@ -73,6 +63,8 @@ export interface ItemListHandle {
|
|||||||
scrollToOffset: (offset: number, options?: { behavior?: 'auto' | 'smooth' }) => void;
|
scrollToOffset: (offset: number, options?: { behavior?: 'auto' | 'smooth' }) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type ItemListItem = Album | AlbumArtist | Artist | Playlist | Song | undefined;
|
||||||
|
|
||||||
export interface ItemListTableComponentProps<TQuery> extends ItemListComponentProps<TQuery> {
|
export interface ItemListTableComponentProps<TQuery> extends ItemListComponentProps<TQuery> {
|
||||||
columns: ItemTableListColumnConfig[];
|
columns: ItemTableListColumnConfig[];
|
||||||
enableAlternateRowColors?: boolean;
|
enableAlternateRowColors?: boolean;
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ export const ExpandedAlbumListItem = ({ item }: ExpandedAlbumListItemProps) => {
|
|||||||
const { data, isLoading } = useSuspenseQuery(
|
const { data, isLoading } = useSuspenseQuery(
|
||||||
albumQueries.detail({
|
albumQueries.detail({
|
||||||
query: { id: item.id },
|
query: { id: item.id },
|
||||||
serverId: item.serverId,
|
serverId: item._serverId,
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@@ -1140,4 +1140,7 @@ export const migrateSettings = (settings: SettingsState, settingsVersion: number
|
|||||||
useSettingsStore.persist.getOptions().migrate!(settings, settingsVersion) as SettingsState;
|
useSettingsStore.persist.getOptions().migrate!(settings, settingsVersion) as SettingsState;
|
||||||
|
|
||||||
export const useListSettings = (type: ItemListKey) =>
|
export const useListSettings = (type: ItemListKey) =>
|
||||||
useSettingsStore((state) => state.lists[type as keyof typeof state.lists], shallow);
|
useSettingsStore(
|
||||||
|
(state) => state.lists[type as keyof typeof state.lists],
|
||||||
|
shallow,
|
||||||
|
) as ItemListSettings;
|
||||||
|
|||||||
Reference in New Issue
Block a user