mirror of
https://github.com/immich-app/immich.git
synced 2025-07-31 15:08:44 -04:00
refactor: asset-viewer/actions
This commit is contained in:
parent
db68d1af9b
commit
5ea66e88f1
@ -3,17 +3,16 @@
|
|||||||
import { page } from '$app/stores';
|
import { page } from '$app/stores';
|
||||||
import { resizeObserver, type OnResizeCallback } from '$lib/actions/resize-observer';
|
import { resizeObserver, type OnResizeCallback } from '$lib/actions/resize-observer';
|
||||||
import { shortcuts, type ShortcutOptions } from '$lib/actions/shortcut';
|
import { shortcuts, type ShortcutOptions } from '$lib/actions/shortcut';
|
||||||
import type { Action } from '$lib/components/asset-viewer/actions/action';
|
|
||||||
import {
|
import {
|
||||||
setFocusToAsset as setFocusAssetInit,
|
setFocusToAsset as setFocusAssetInit,
|
||||||
setFocusTo as setFocusToInit,
|
setFocusTo as setFocusToInit,
|
||||||
} from '$lib/components/photos-page/actions/focus-actions';
|
} from '$lib/components/photos-page/actions/focus-actions';
|
||||||
|
import AssetViewerAndActions from '$lib/components/photos-page/asset-viewer-and-actions.svelte';
|
||||||
import Skeleton from '$lib/components/photos-page/skeleton.svelte';
|
import Skeleton from '$lib/components/photos-page/skeleton.svelte';
|
||||||
import ChangeDate from '$lib/components/shared-components/change-date.svelte';
|
import ChangeDate from '$lib/components/shared-components/change-date.svelte';
|
||||||
import Scrubber from '$lib/components/shared-components/scrubber/scrubber.svelte';
|
import Scrubber from '$lib/components/shared-components/scrubber/scrubber.svelte';
|
||||||
import { AppRoute, AssetAction } from '$lib/constants';
|
import { AppRoute, AssetAction } from '$lib/constants';
|
||||||
import { albumMapViewManager } from '$lib/managers/album-view-map.manager.svelte';
|
import { albumMapViewManager } from '$lib/managers/album-view-map.manager.svelte';
|
||||||
import { authManager } from '$lib/managers/auth-manager.svelte';
|
|
||||||
import { modalManager } from '$lib/managers/modal-manager.svelte';
|
import { modalManager } from '$lib/managers/modal-manager.svelte';
|
||||||
import type { MonthGroup } from '$lib/managers/timeline-manager/month-group.svelte';
|
import type { MonthGroup } from '$lib/managers/timeline-manager/month-group.svelte';
|
||||||
import { TimelineManager } from '$lib/managers/timeline-manager/timeline-manager.svelte';
|
import { TimelineManager } from '$lib/managers/timeline-manager/timeline-manager.svelte';
|
||||||
@ -28,11 +27,11 @@
|
|||||||
import { searchStore } from '$lib/stores/search.svelte';
|
import { searchStore } from '$lib/stores/search.svelte';
|
||||||
import { featureFlags } from '$lib/stores/server-config.store';
|
import { featureFlags } from '$lib/stores/server-config.store';
|
||||||
import { handlePromiseError } from '$lib/utils';
|
import { handlePromiseError } from '$lib/utils';
|
||||||
import { deleteAssets, updateStackedAssetInTimeline, updateUnstackedAssetInTimeline } from '$lib/utils/actions';
|
import { deleteAssets, updateStackedAssetInTimeline } from '$lib/utils/actions';
|
||||||
import { archiveAssets, cancelMultiselect, selectAllAssets, stackAssets } from '$lib/utils/asset-utils';
|
import { archiveAssets, cancelMultiselect, selectAllAssets, stackAssets } from '$lib/utils/asset-utils';
|
||||||
import { navigate } from '$lib/utils/navigation';
|
import { navigate } from '$lib/utils/navigation';
|
||||||
import { toTimelineAsset, type ScrubberListener, type TimelinePlainYearMonth } from '$lib/utils/timeline-util';
|
import { type ScrubberListener, type TimelinePlainYearMonth } from '$lib/utils/timeline-util';
|
||||||
import { AssetVisibility, getAssetInfo, type AlbumResponseDto, type PersonResponseDto } from '@immich/sdk';
|
import { AssetVisibility, type AlbumResponseDto, type PersonResponseDto } from '@immich/sdk';
|
||||||
import { DateTime } from 'luxon';
|
import { DateTime } from 'luxon';
|
||||||
import { onMount, type Snippet } from 'svelte';
|
import { onMount, type Snippet } from 'svelte';
|
||||||
import type { UpdatePayload } from 'vite';
|
import type { UpdatePayload } from 'vite';
|
||||||
@ -54,8 +53,7 @@
|
|||||||
| AssetAction.ARCHIVE
|
| AssetAction.ARCHIVE
|
||||||
| AssetAction.FAVORITE
|
| AssetAction.FAVORITE
|
||||||
| AssetAction.UNFAVORITE
|
| AssetAction.UNFAVORITE
|
||||||
| AssetAction.SET_VISIBILITY_TIMELINE
|
| AssetAction.SET_VISIBILITY_TIMELINE;
|
||||||
| null;
|
|
||||||
withStacked?: boolean;
|
withStacked?: boolean;
|
||||||
showArchiveIcon?: boolean;
|
showArchiveIcon?: boolean;
|
||||||
isShared?: boolean;
|
isShared?: boolean;
|
||||||
@ -74,7 +72,7 @@
|
|||||||
enableRouting,
|
enableRouting,
|
||||||
timelineManager = $bindable(),
|
timelineManager = $bindable(),
|
||||||
assetInteraction,
|
assetInteraction,
|
||||||
removeAction = null,
|
removeAction,
|
||||||
withStacked = false,
|
withStacked = false,
|
||||||
showArchiveIcon = false,
|
showArchiveIcon = false,
|
||||||
isShared = false,
|
isShared = false,
|
||||||
@ -87,7 +85,7 @@
|
|||||||
empty,
|
empty,
|
||||||
}: Props = $props();
|
}: Props = $props();
|
||||||
|
|
||||||
let { isViewing: showAssetViewer, asset: viewingAsset, preloadAssets, gridScrollTarget } = assetViewingStore;
|
let { isViewing: showAssetViewer, gridScrollTarget } = assetViewingStore;
|
||||||
|
|
||||||
let element: HTMLElement | undefined = $state();
|
let element: HTMLElement | undefined = $state();
|
||||||
|
|
||||||
@ -433,104 +431,6 @@
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handlePrevious = async () => {
|
|
||||||
const laterAsset = await timelineManager.getLaterAsset($viewingAsset);
|
|
||||||
|
|
||||||
if (laterAsset) {
|
|
||||||
const preloadAsset = await timelineManager.getLaterAsset(laterAsset);
|
|
||||||
const asset = await getAssetInfo({ id: laterAsset.id, key: authManager.key });
|
|
||||||
assetViewingStore.setAsset(asset, preloadAsset ? [preloadAsset] : []);
|
|
||||||
await navigate({ targetRoute: 'current', assetId: laterAsset.id });
|
|
||||||
}
|
|
||||||
|
|
||||||
return !!laterAsset;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleNext = async () => {
|
|
||||||
const earlierAsset = await timelineManager.getEarlierAsset($viewingAsset);
|
|
||||||
if (earlierAsset) {
|
|
||||||
const preloadAsset = await timelineManager.getEarlierAsset(earlierAsset);
|
|
||||||
const asset = await getAssetInfo({ id: earlierAsset.id, key: authManager.key });
|
|
||||||
assetViewingStore.setAsset(asset, preloadAsset ? [preloadAsset] : []);
|
|
||||||
await navigate({ targetRoute: 'current', assetId: earlierAsset.id });
|
|
||||||
}
|
|
||||||
|
|
||||||
return !!earlierAsset;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleRandom = async () => {
|
|
||||||
const randomAsset = await timelineManager.getRandomAsset();
|
|
||||||
|
|
||||||
if (randomAsset) {
|
|
||||||
const asset = await getAssetInfo({ id: randomAsset.id, key: authManager.key });
|
|
||||||
assetViewingStore.setAsset(asset);
|
|
||||||
await navigate({ targetRoute: 'current', assetId: randomAsset.id });
|
|
||||||
return asset;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClose = async (asset: { id: string }) => {
|
|
||||||
assetViewingStore.showAssetViewer(false);
|
|
||||||
showSkeleton = true;
|
|
||||||
$gridScrollTarget = { at: asset.id };
|
|
||||||
await navigate({ targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget });
|
|
||||||
};
|
|
||||||
|
|
||||||
const handlePreAction = async (action: Action) => {
|
|
||||||
switch (action.type) {
|
|
||||||
case removeAction:
|
|
||||||
case AssetAction.TRASH:
|
|
||||||
case AssetAction.RESTORE:
|
|
||||||
case AssetAction.DELETE:
|
|
||||||
case AssetAction.ARCHIVE:
|
|
||||||
case AssetAction.SET_VISIBILITY_LOCKED:
|
|
||||||
case AssetAction.SET_VISIBILITY_TIMELINE: {
|
|
||||||
// find the next asset to show or close the viewer
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unused-expressions
|
|
||||||
(await handleNext()) || (await handlePrevious()) || (await handleClose(action.asset));
|
|
||||||
|
|
||||||
// delete after find the next one
|
|
||||||
timelineManager.removeAssets([action.asset.id]);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const handleAction = (action: Action) => {
|
|
||||||
switch (action.type) {
|
|
||||||
case AssetAction.ARCHIVE:
|
|
||||||
case AssetAction.UNARCHIVE:
|
|
||||||
case AssetAction.FAVORITE:
|
|
||||||
case AssetAction.UNFAVORITE: {
|
|
||||||
timelineManager.updateAssets([action.asset]);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case AssetAction.ADD: {
|
|
||||||
timelineManager.addAssets([action.asset]);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
case AssetAction.UNSTACK: {
|
|
||||||
updateUnstackedAssetInTimeline(timelineManager, action.assets);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case AssetAction.SET_STACK_PRIMARY_ASSET: {
|
|
||||||
//Have to unstack then restack assets in timeline in order for the currently removed new primary asset to be made visible.
|
|
||||||
updateUnstackedAssetInTimeline(
|
|
||||||
timelineManager,
|
|
||||||
action.stack.assets.map((asset) => toTimelineAsset(asset)),
|
|
||||||
);
|
|
||||||
updateStackedAssetInTimeline(timelineManager, {
|
|
||||||
stack: action.stack,
|
|
||||||
toDeleteIds: action.stack.assets
|
|
||||||
.filter((asset) => asset.id !== action.stack.primaryAssetId)
|
|
||||||
.map((asset) => asset.id),
|
|
||||||
});
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let lastAssetMouseEvent: TimelineAsset | null = $state(null);
|
let lastAssetMouseEvent: TimelineAsset | null = $state(null);
|
||||||
|
|
||||||
let shiftKeyIsDown = $state(false);
|
let shiftKeyIsDown = $state(false);
|
||||||
@ -703,7 +603,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
isShortcutModalOpen = true;
|
isShortcutModalOpen = true;
|
||||||
await modalManager.show(ShortcutsModal);
|
await modalManager.show(ShortcutsModal, {});
|
||||||
isShortcutModalOpen = false;
|
isShortcutModalOpen = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -914,22 +814,16 @@
|
|||||||
{#if !albumMapViewManager.isInMapView}
|
{#if !albumMapViewManager.isInMapView}
|
||||||
<Portal target="body">
|
<Portal target="body">
|
||||||
{#if $showAssetViewer}
|
{#if $showAssetViewer}
|
||||||
{#await import('../asset-viewer/asset-viewer.svelte') then { default: AssetViewer }}
|
<AssetViewerAndActions
|
||||||
<AssetViewer
|
bind:showSkeleton
|
||||||
{withStacked}
|
{timelineManager}
|
||||||
asset={$viewingAsset}
|
{removeAction}
|
||||||
preloadAssets={$preloadAssets}
|
{withStacked}
|
||||||
{isShared}
|
{isShared}
|
||||||
{album}
|
{album}
|
||||||
{person}
|
{person}
|
||||||
preAction={handlePreAction}
|
{isShowDeleteConfirmation}
|
||||||
onAction={handleAction}
|
></AssetViewerAndActions>
|
||||||
onPrevious={handlePrevious}
|
|
||||||
onNext={handleNext}
|
|
||||||
onRandom={handleRandom}
|
|
||||||
onClose={handleClose}
|
|
||||||
/>
|
|
||||||
{/await}
|
|
||||||
{/if}
|
{/if}
|
||||||
</Portal>
|
</Portal>
|
||||||
{/if}
|
{/if}
|
||||||
|
140
web/src/lib/components/photos-page/asset-viewer-actions.svelte
Normal file
140
web/src/lib/components/photos-page/asset-viewer-actions.svelte
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import type { Action } from '$lib/components/asset-viewer/actions/action';
|
||||||
|
import { AssetAction } from '$lib/constants';
|
||||||
|
import { authManager } from '$lib/managers/auth-manager.svelte';
|
||||||
|
import { TimelineManager } from '$lib/managers/timeline-manager/timeline-manager.svelte';
|
||||||
|
import { assetViewingStore } from '$lib/stores/asset-viewing.store';
|
||||||
|
import { updateStackedAssetInTimeline, updateUnstackedAssetInTimeline } from '$lib/utils/actions';
|
||||||
|
import { navigate } from '$lib/utils/navigation';
|
||||||
|
import { toTimelineAsset } from '$lib/utils/timeline-util';
|
||||||
|
import { getAssetInfo, type AssetResponseDto } from '@immich/sdk';
|
||||||
|
|
||||||
|
let { asset: viewingAsset, gridScrollTarget } = assetViewingStore;
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
timelineManager: TimelineManager;
|
||||||
|
showSkeleton: boolean;
|
||||||
|
removeAction?:
|
||||||
|
| AssetAction.UNARCHIVE
|
||||||
|
| AssetAction.ARCHIVE
|
||||||
|
| AssetAction.FAVORITE
|
||||||
|
| AssetAction.UNFAVORITE
|
||||||
|
| AssetAction.SET_VISIBILITY_TIMELINE;
|
||||||
|
handlePreAction?: (action: Action) => Promise<void>;
|
||||||
|
handleAction?: (action: Action) => void;
|
||||||
|
handleNext?: () => Promise<boolean>;
|
||||||
|
handlePrevious?: () => Promise<boolean>;
|
||||||
|
handleRandom?: () => Promise<AssetResponseDto | undefined>;
|
||||||
|
handleClose?: (asset: { id: string }) => Promise<void>;
|
||||||
|
}
|
||||||
|
|
||||||
|
let {
|
||||||
|
timelineManager = $bindable(),
|
||||||
|
showSkeleton = $bindable(false),
|
||||||
|
removeAction,
|
||||||
|
handlePreAction = $bindable(),
|
||||||
|
handleAction = $bindable(),
|
||||||
|
handleNext = $bindable(),
|
||||||
|
handlePrevious = $bindable(),
|
||||||
|
handleRandom = $bindable(),
|
||||||
|
handleClose = $bindable(),
|
||||||
|
}: Props = $props();
|
||||||
|
|
||||||
|
handlePrevious = async () => {
|
||||||
|
const laterAsset = await timelineManager.getLaterAsset($viewingAsset);
|
||||||
|
|
||||||
|
if (laterAsset) {
|
||||||
|
const preloadAsset = await timelineManager.getLaterAsset(laterAsset);
|
||||||
|
const asset = await getAssetInfo({ id: laterAsset.id, key: authManager.key });
|
||||||
|
assetViewingStore.setAsset(asset, preloadAsset ? [preloadAsset] : []);
|
||||||
|
await navigate({ targetRoute: 'current', assetId: laterAsset.id });
|
||||||
|
}
|
||||||
|
|
||||||
|
return !!laterAsset;
|
||||||
|
};
|
||||||
|
|
||||||
|
handleNext = async () => {
|
||||||
|
const earlierAsset = await timelineManager.getEarlierAsset($viewingAsset);
|
||||||
|
if (earlierAsset) {
|
||||||
|
const preloadAsset = await timelineManager.getEarlierAsset(earlierAsset);
|
||||||
|
const asset = await getAssetInfo({ id: earlierAsset.id, key: authManager.key });
|
||||||
|
assetViewingStore.setAsset(asset, preloadAsset ? [preloadAsset] : []);
|
||||||
|
await navigate({ targetRoute: 'current', assetId: earlierAsset.id });
|
||||||
|
}
|
||||||
|
|
||||||
|
return !!earlierAsset;
|
||||||
|
};
|
||||||
|
|
||||||
|
handleRandom = async () => {
|
||||||
|
const randomAsset = await timelineManager.getRandomAsset();
|
||||||
|
|
||||||
|
if (randomAsset) {
|
||||||
|
const asset = await getAssetInfo({ id: randomAsset.id, key: authManager.key });
|
||||||
|
assetViewingStore.setAsset(asset);
|
||||||
|
await navigate({ targetRoute: 'current', assetId: randomAsset.id });
|
||||||
|
return asset;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
handleClose = async (asset: { id: string }) => {
|
||||||
|
assetViewingStore.showAssetViewer(false);
|
||||||
|
showSkeleton = true;
|
||||||
|
$gridScrollTarget = { at: asset.id };
|
||||||
|
await navigate({ targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget });
|
||||||
|
};
|
||||||
|
|
||||||
|
handlePreAction = async (action: Action) => {
|
||||||
|
switch (action.type) {
|
||||||
|
case removeAction:
|
||||||
|
case AssetAction.TRASH:
|
||||||
|
case AssetAction.RESTORE:
|
||||||
|
case AssetAction.DELETE:
|
||||||
|
case AssetAction.ARCHIVE:
|
||||||
|
case AssetAction.SET_VISIBILITY_LOCKED:
|
||||||
|
case AssetAction.SET_VISIBILITY_TIMELINE: {
|
||||||
|
// find the next asset to show or close the viewer
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-expressions
|
||||||
|
(await handleNext()) || (await handlePrevious()) || (await handleClose(action.asset));
|
||||||
|
|
||||||
|
// delete after find the next one
|
||||||
|
timelineManager.removeAssets([action.asset.id]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
handleAction = (action: Action) => {
|
||||||
|
switch (action.type) {
|
||||||
|
case AssetAction.ARCHIVE:
|
||||||
|
case AssetAction.UNARCHIVE:
|
||||||
|
case AssetAction.FAVORITE:
|
||||||
|
case AssetAction.UNFAVORITE: {
|
||||||
|
timelineManager.updateAssets([action.asset]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AssetAction.ADD: {
|
||||||
|
timelineManager.addAssets([action.asset]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AssetAction.UNSTACK: {
|
||||||
|
updateUnstackedAssetInTimeline(timelineManager, action.assets);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AssetAction.SET_STACK_PRIMARY_ASSET: {
|
||||||
|
//Have to unstack then restack assets in timeline in order for the currently removed new primary asset to be made visible.
|
||||||
|
updateUnstackedAssetInTimeline(
|
||||||
|
timelineManager,
|
||||||
|
action.stack.assets.map((asset) => toTimelineAsset(asset)),
|
||||||
|
);
|
||||||
|
updateStackedAssetInTimeline(timelineManager, {
|
||||||
|
stack: action.stack,
|
||||||
|
toDeleteIds: action.stack.assets
|
||||||
|
.filter((asset) => asset.id !== action.stack.primaryAssetId)
|
||||||
|
.map((asset) => asset.id),
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
@ -0,0 +1,73 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import type { Action } from '$lib/components/asset-viewer/actions/action';
|
||||||
|
import AssetViewerActions from '$lib/components/photos-page/asset-viewer-actions.svelte';
|
||||||
|
import { AssetAction } from '$lib/constants';
|
||||||
|
import { TimelineManager } from '$lib/managers/timeline-manager/timeline-manager.svelte';
|
||||||
|
import { assetViewingStore } from '$lib/stores/asset-viewing.store';
|
||||||
|
import { type AlbumResponseDto, type AssetResponseDto, type PersonResponseDto } from '@immich/sdk';
|
||||||
|
|
||||||
|
let { asset: viewingAsset, preloadAssets } = assetViewingStore;
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
timelineManager: TimelineManager;
|
||||||
|
showSkeleton: boolean;
|
||||||
|
removeAction?:
|
||||||
|
| AssetAction.UNARCHIVE
|
||||||
|
| AssetAction.ARCHIVE
|
||||||
|
| AssetAction.FAVORITE
|
||||||
|
| AssetAction.UNFAVORITE
|
||||||
|
| AssetAction.SET_VISIBILITY_TIMELINE;
|
||||||
|
withStacked?: boolean;
|
||||||
|
isShared?: boolean;
|
||||||
|
album?: AlbumResponseDto | null;
|
||||||
|
person?: PersonResponseDto | null;
|
||||||
|
isShowDeleteConfirmation?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
let {
|
||||||
|
timelineManager = $bindable(),
|
||||||
|
showSkeleton = $bindable(false),
|
||||||
|
removeAction,
|
||||||
|
withStacked = false,
|
||||||
|
isShared = false,
|
||||||
|
album = null,
|
||||||
|
person = null,
|
||||||
|
isShowDeleteConfirmation = $bindable(false),
|
||||||
|
}: Props = $props();
|
||||||
|
|
||||||
|
let handlePreAction = <(action: Action) => Promise<void>>$state();
|
||||||
|
let handleAction = <(action: Action) => void>$state();
|
||||||
|
let handleNext = <() => Promise<boolean>>$state();
|
||||||
|
let handlePrevious = <() => Promise<boolean>>$state();
|
||||||
|
let handleRandom = <() => Promise<AssetResponseDto | undefined>>$state();
|
||||||
|
let handleClose = <(asset: { id: string }) => Promise<void>>$state();
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<AssetViewerActions
|
||||||
|
{timelineManager}
|
||||||
|
{removeAction}
|
||||||
|
bind:showSkeleton
|
||||||
|
bind:handlePreAction
|
||||||
|
bind:handleAction
|
||||||
|
bind:handleNext
|
||||||
|
bind:handlePrevious
|
||||||
|
bind:handleRandom
|
||||||
|
bind:handleClose
|
||||||
|
></AssetViewerActions>
|
||||||
|
|
||||||
|
{#await import('../asset-viewer/asset-viewer.svelte') then { default: AssetViewer }}
|
||||||
|
<AssetViewer
|
||||||
|
{withStacked}
|
||||||
|
asset={$viewingAsset}
|
||||||
|
preloadAssets={$preloadAssets}
|
||||||
|
{isShared}
|
||||||
|
{album}
|
||||||
|
{person}
|
||||||
|
preAction={handlePreAction}
|
||||||
|
onAction={handleAction}
|
||||||
|
onPrevious={handlePrevious}
|
||||||
|
onNext={handleNext}
|
||||||
|
onRandom={handleRandom}
|
||||||
|
onClose={handleClose}
|
||||||
|
/>
|
||||||
|
{/await}
|
Loading…
x
Reference in New Issue
Block a user