add item list selection dialog

This commit is contained in:
jeffvli
2025-12-16 20:24:46 -08:00
parent c9cd87bae5
commit e21f538aa4
8 changed files with 173 additions and 1 deletions
+4
View File
@@ -2,11 +2,14 @@
"action": {
"addToFavorites": "add to $t(entity.favorite_other)",
"addToPlaylist": "add to $t(entity.playlist_one)",
"addOrRemoveFromSelection": "add or remove from selection",
"selectRangeOfItems": "select a range of items",
"clearQueue": "clear queue",
"createPlaylist": "create $t(entity.playlist_one)",
"createRadioStation": "create $t(entity.radioStation_one)",
"deletePlaylist": "delete $t(entity.playlist_one)",
"deleteRadioStation": "delete $t(entity.radioStation_one)",
"selectAll": "select all",
"deselectAll": "deselect all",
"downloadStarted": "started download of {{count}} items",
"editPlaylist": "edit $t(entity.playlist_one)",
@@ -37,6 +40,7 @@
}
},
"common": {
"countSelected": "{{count}} selected",
"explicitStatus": "explicit status",
"action_one": "action",
"action_other": "actions",
@@ -1,4 +1,5 @@
.item-grid-container {
position: relative;
display: flex;
flex-direction: column !important;
width: 100%;
@@ -41,6 +41,7 @@ import {
useItemListState,
useItemListStateSubscription,
} from '/@/renderer/components/item-list/helpers/item-list-state';
import { SelectionDialog } from '/@/renderer/components/item-list/selection-dialog';
import { ItemControls, ItemListHandle } from '/@/renderer/components/item-list/types';
import { animationProps } from '/@/shared/components/animations/animation-props';
import { useElementSize } from '/@/shared/hooks/use-element-size';
@@ -742,7 +743,10 @@ const BaseItemGridList = ({
/>
)}
</AutoSizer>
<ExpandedContainer internalState={internalState} itemType={itemType} />
<AnimatePresence presenceAffectsLayout>
<ExpandedContainer internalState={internalState} itemType={itemType} />
<SelectionDialog internalState={internalState} />
</AnimatePresence>
</motion.div>
);
};
@@ -1,4 +1,5 @@
.item-table-list-container {
position: relative;
display: flex;
flex-direction: column;
height: 100%;
@@ -34,6 +34,7 @@ import {
import { parseTableColumns } from '/@/renderer/components/item-list/helpers/parse-table-columns';
import { useStickyTableGroupRows } from '/@/renderer/components/item-list/item-table-list/hooks/use-sticky-table-group-rows';
import { useStickyTableHeader } from '/@/renderer/components/item-list/item-table-list/hooks/use-sticky-table-header';
import { SelectionDialog } from '/@/renderer/components/item-list/selection-dialog';
import {
ItemControls,
ItemListHandle,
@@ -2318,6 +2319,7 @@ const BaseItemTableList = ({
totalRowCount={totalRowCount}
/>
<ExpandedContainer internalState={internalState} itemType={itemType} />
<SelectionDialog internalState={internalState} />
</motion.div>
);
};
@@ -0,0 +1,22 @@
.selection-indicator {
position: absolute;
bottom: 0;
left: 50%;
z-index: 100;
min-width: 320px;
padding: var(--theme-spacing-sm) var(--theme-spacing-md);
color: var(--theme-colors-surface-foreground);
background: color-mix(in srgb, var(--theme-colors-surface) 85%, transparent);
border: 1px solid color-mix(in srgb, var(--theme-colors-border) 50%, transparent);
border-radius: var(--theme-radius-md);
box-shadow:
2px 2px 10px 2px rgb(0 0 0 / 40%),
0 0 0 1px rgb(255 255 255 / 5%);
backdrop-filter: blur(12px) saturate(180%);
transform: translateX(-50%);
}
.info-icon {
display: flex;
cursor: pointer;
}
@@ -0,0 +1,135 @@
import { AnimatePresence, motion } from 'motion/react';
import { useTranslation } from 'react-i18next';
import styles from './selection-dialog.module.css';
import i18n from '/@/i18n/i18n';
import {
ItemListStateActions,
useItemListStateSubscription,
} from '/@/renderer/components/item-list/helpers/item-list-state';
import { ContextMenuController } from '/@/renderer/features/context-menu/context-menu-controller';
import { ActionIcon } from '/@/shared/components/action-icon/action-icon';
import { animationProps } from '/@/shared/components/animations/animation-props';
import { Group } from '/@/shared/components/group/group';
import { HoverCard } from '/@/shared/components/hover-card/hover-card';
import { Icon } from '/@/shared/components/icon/icon';
import { Kbd } from '/@/shared/components/kbd/kbd';
import { Table } from '/@/shared/components/table/table';
import { Text } from '/@/shared/components/text/text';
const controls = [
{
control1: <Kbd>CTRL</Kbd>,
control2: <Kbd>A</Kbd>,
label: i18n.t('action.selectAll', { postProcess: 'sentenceCase' }),
},
{
control1: <Kbd>CTRL</Kbd>,
control2: <Icon fill="default" icon="mouseLeftClick" />,
label: i18n.t('action.addOrRemoveFromSelection', { postProcess: 'sentenceCase' }),
},
{
control1: <Kbd>SHIFT</Kbd>,
control2: <Icon fill="default" icon="mouseLeftClick" />,
label: i18n.t('action.selectRangeOfItems', { postProcess: 'sentenceCase' }),
},
];
export const SelectionDialog = ({ internalState }: { internalState: ItemListStateActions }) => {
const { t } = useTranslation();
const isListExpanded = useItemListStateSubscription(internalState, (state) =>
state ? state.expanded.size > 0 : false,
);
const selectedCount = useItemListStateSubscription(internalState, (state) =>
state ? state.selected.size : 0,
);
const handleClearSelection = () => {
internalState.clearSelected();
};
const handleOpenMoreActions = (event: React.MouseEvent<unknown>) => {
event.preventDefault();
event.stopPropagation();
const selectedItems = internalState.getSelected();
if (selectedItems.length === 0) {
return;
}
ContextMenuController.call({
cmd: { items: selectedItems as any[], type: (selectedItems[0] as any)._itemType },
event,
});
};
const isOpen = selectedCount > 0;
return (
<AnimatePresence initial={false} mode="sync">
{isOpen && (
<motion.div
{...animationProps.fadeIn}
className={styles.selectionIndicator}
style={{ bottom: isListExpanded ? '320px' : '1rem' }}
>
<Group gap="xl" justify="space-between">
<Group gap="sm">
<HoverCard offset={20} position="top">
<HoverCard.Target>
<span className={styles.infoIcon}>
<Icon icon="keyboard" />
</span>
</HoverCard.Target>
<HoverCard.Dropdown>
<Table>
<Table.Tbody>
{controls.map((control) => (
<Table.Tr key={control.label}>
<Table.Td ta="start">
{control.control1}
</Table.Td>
<Table.Td>+</Table.Td>
<Table.Td ta="center">
{control.control2}
</Table.Td>
<Table.Td>
<Text size="xs">{control.label}</Text>
</Table.Td>
</Table.Tr>
))}
</Table.Tbody>
</Table>
</HoverCard.Dropdown>
</HoverCard>
<Text fw={500} isNoSelect size="sm">
{t('common.countSelected', { count: selectedCount })}
</Text>
</Group>
<Group gap="xs">
<ActionIcon
icon="x"
iconProps={{ size: 'xl' }}
onClick={handleClearSelection}
size="xs"
variant="subtle"
/>
<ActionIcon
icon="ellipsisHorizontal"
iconProps={{ size: 'xl' }}
onClick={handleOpenMoreActions}
size="xs"
variant="subtle"
/>
</Group>
</Group>
</motion.div>
)}
</AnimatePresence>
);
};
+3
View File
@@ -114,6 +114,7 @@ import {
LuX,
} from 'react-icons/lu';
import { MdOutlineVisibility, MdOutlineVisibilityOff } from 'react-icons/md';
import { PiMouseLeftClickFill, PiMouseRightClickFill } from 'react-icons/pi';
import { RiPlayListAddLine, RiRepeat2Line, RiRepeatOneLine } from 'react-icons/ri';
import { SiMusicbrainz } from 'react-icons/si';
@@ -202,6 +203,8 @@ export const AppIcon = {
metadata: LuBookOpen,
microphone: LuMicVocal,
minus: LuMinus,
mouseLeftClick: PiMouseLeftClickFill,
mouseRightClick: PiMouseRightClickFill,
panelRightClose: LuPanelRightClose,
panelRightOpen: LuPanelRightOpen,
pin: LuPin,