import { Divider, Stack } from '@mantine/core'; import { useClickOutside, useResizeObserver, useSetState, useViewportSize } from '@mantine/hooks'; import { createContext, Fragment, useState } from 'react'; import { ContextMenu, ContextMenuButton } from '/@/renderer/components'; import { OpenContextMenuProps, SetContextMenuItems, useContextMenuEvents, } from '/@/renderer/features/context-menu/events'; import { useHandlePlayQueueAdd } from '/@/renderer/features/player/hooks/use-handle-playqueue-add'; import { LibraryItem, Play } from '/@/renderer/types'; type ContextMenuContextProps = { closeContextMenu: () => void; openContextMenu: (args: OpenContextMenuProps) => void; }; const ContextMenuContext = createContext({ closeContextMenu: () => {}, openContextMenu: (args: OpenContextMenuProps) => { return args; }, }); export interface ContextMenuProviderProps { children: React.ReactNode; } export const ContextMenuProvider = ({ children }: ContextMenuProviderProps) => { const [opened, setOpened] = useState(false); const clickOutsideRef = useClickOutside(() => setOpened(false)); const viewport = useViewportSize(); const [ref, menuRect] = useResizeObserver(); const [ctx, setCtx] = useSetState<{ data: any[]; menuItems: SetContextMenuItems; type: LibraryItem; xPos: number; yPos: number; }>({ data: [], menuItems: [], type: LibraryItem.SONG, xPos: 0, yPos: 0, }); const handlePlayQueueAdd = useHandlePlayQueueAdd(); const openContextMenu = (args: OpenContextMenuProps) => { const { xPos, yPos, menuItems, data, type } = args; const shouldReverseY = yPos + menuRect.height > viewport.height; const shouldReverseX = xPos + menuRect.width > viewport.width; const calculatedXPos = shouldReverseX ? xPos - menuRect.width : xPos; const calculatedYPos = shouldReverseY ? yPos - menuRect.height : yPos; setCtx({ data, menuItems, type, xPos: calculatedXPos, yPos: calculatedYPos }); setOpened(true); }; const closeContextMenu = () => { setOpened(false); }; useContextMenuEvents({ closeContextMenu, openContextMenu, }); const handlePlay = (play: Play) => { console.log('ctx', ctx); switch (ctx.type) { case LibraryItem.ALBUM: handlePlayQueueAdd({ byItemType: { id: ctx.data.map((item) => item.id), type: ctx.type }, play, }); break; case LibraryItem.ARTIST: handlePlayQueueAdd({ byItemType: { id: ctx.data.map((item) => item.id), type: ctx.type }, play, }); break; case LibraryItem.ALBUM_ARTIST: handlePlayQueueAdd({ byItemType: { id: ctx.data.map((item) => item.id), type: ctx.type }, play, }); break; case LibraryItem.SONG: handlePlayQueueAdd({ byData: ctx.data, play }); break; case LibraryItem.PLAYLIST: handlePlayQueueAdd({ byItemType: { id: ctx.data.map((item) => item.id), type: ctx.type }, play, }); break; } }; const contextMenuItems = { addToFavorites: { id: 'addToFavorites', label: 'Add to favorites', onClick: () => {} }, addToPlaylist: { id: 'addToPlaylist', label: 'Add to playlist', onClick: () => {} }, deletePlaylist: { id: 'deletePlaylist', label: 'Delete playlist', onClick: () => {} }, play: { id: 'play', label: 'Play', onClick: () => handlePlay(Play.NOW), }, playLast: { id: 'playLast', label: 'Add to queue (last)', onClick: () => handlePlay(Play.LAST), }, playNext: { id: 'playNext', label: 'Add to queue (next)', onClick: () => handlePlay(Play.NEXT), }, removeFromFavorites: { id: 'removeFromFavorites', label: 'Remove from favorites', onClick: () => {}, }, setRating: { id: 'setRating', label: 'Set rating', onClick: () => {} }, }; return ( {opened && ( {ctx.menuItems?.map((item) => { return ( {contextMenuItems[item.id as keyof typeof contextMenuItems].label} {item.divider && } ); })} )} {children} ); };