import React, { createRef, useEffect, useState } from 'react'; import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import { faEllipsisH } from '@fortawesome/free-solid-svg-icons/faEllipsisH'; import { CSSTransition } from 'react-transition-group'; import { faPencilAlt } from '@fortawesome/free-solid-svg-icons/faPencilAlt'; import { faTrashAlt } from '@fortawesome/free-solid-svg-icons/faTrashAlt'; import { faFileDownload } from '@fortawesome/free-solid-svg-icons/faFileDownload'; import { faCopy } from '@fortawesome/free-solid-svg-icons/faCopy'; import { faLevelUpAlt } from '@fortawesome/free-solid-svg-icons/faLevelUpAlt'; import RenameFileModal from '@/components/server/files/RenameFileModal'; import { ServerContext } from '@/state/server'; import { join } from 'path'; import deleteFile from '@/api/server/files/deleteFile'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import copyFile from '@/api/server/files/copyFile'; import { httpErrorToHuman } from '@/api/http'; import Can from '@/components/elements/Can'; import getFileDownloadUrl from '@/api/server/files/getFileDownloadUrl'; import useServer from '@/plugins/useServer'; import useFlash from '@/plugins/useFlash'; import tw from 'twin.macro'; import Fade from '@/components/elements/Fade'; type ModalType = 'rename' | 'move'; export default ({ uuid }: { uuid: string }) => { const menu = createRef(); const menuButton = createRef(); const [ menuVisible, setMenuVisible ] = useState(false); const [ showSpinner, setShowSpinner ] = useState(false); const [ modal, setModal ] = useState(null); const [ posX, setPosX ] = useState(0); const server = useServer(); const { addError, clearFlashes } = useFlash(); const file = ServerContext.useStoreState(state => state.files.contents.find(file => file.uuid === uuid)); const directory = ServerContext.useStoreState(state => state.files.directory); const { removeFile, getDirectoryContents } = ServerContext.useStoreActions(actions => actions.files); if (!file) { return null; } const windowListener = (e: MouseEvent) => { if (e.button === 2 || !menuVisible || !menu.current) { return; } if (e.target === menu.current || menu.current.contains(e.target as Node)) { return; } if (e.target !== menu.current && !menu.current.contains(e.target as Node)) { setMenuVisible(false); } }; const doDeletion = () => { setShowSpinner(true); clearFlashes('files'); deleteFile(server.uuid, join(directory, file.name)) .then(() => removeFile(uuid)) .catch(error => { console.error('Error while attempting to delete a file.', error); addError({ key: 'files', message: httpErrorToHuman(error) }); setShowSpinner(false); }); }; const doCopy = () => { setShowSpinner(true); clearFlashes('files'); copyFile(server.uuid, join(directory, file.name)) .then(() => getDirectoryContents(directory)) .catch(error => { console.error('Error while attempting to copy file.', error); addError({ key: 'files', message: httpErrorToHuman(error) }); setShowSpinner(false); }); }; const doDownload = () => { setShowSpinner(true); clearFlashes('files'); getFileDownloadUrl(server.uuid, join(directory, file.name)) .then(url => { // @ts-ignore window.location = url; }) .catch(error => { console.error(error); addError({ key: 'files', message: httpErrorToHuman(error) }); }) .then(() => setShowSpinner(false)); }; useEffect(() => { menuVisible ? document.addEventListener('click', windowListener) : document.removeEventListener('click', windowListener); if (menuVisible && menu.current) { menu.current.setAttribute( 'style', `margin-top: -0.35rem; left: ${Math.round(posX - menu.current.clientWidth)}px`, ); } }, [ menuVisible ]); useEffect(() => () => { document.removeEventListener('click', windowListener); }, []); return (
{ e.preventDefault(); if (!menuVisible) { setPosX(e.clientX); } setModal(null); setMenuVisible(!menuVisible); }} > { setModal(null); setMenuVisible(false); }} />
{ e.stopPropagation(); setMenuVisible(false); }} css={tw`absolute bg-white p-2 rounded border border-neutral-700 shadow-lg text-neutral-500 min-w-48`} >
setModal('rename')} css={tw`hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded`} > Rename
setModal('move')} css={tw`hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded`} > Move
doCopy()} css={tw`hover:text-neutral-700 p-2 flex items-center hover:bg-neutral-100 rounded`} > Copy
doDownload()} > Download
doDeletion()} css={tw`hover:text-red-700 p-2 flex items-center hover:bg-red-100 rounded`} > Delete
); };