204 lines
7.3 KiB
TypeScript
204 lines
7.3 KiB
TypeScript
import { FunctionComponent, useCallback, useMemo } from 'react'
|
|
import { FileItemActionType } from '../AttachedFilesPopover/PopoverFileItemAction'
|
|
import Icon from '@/Components/Icon/Icon'
|
|
import { observer } from 'mobx-react-lite'
|
|
import { formatSizeToReadableString } from '@standardnotes/filepicker'
|
|
import { useResponsiveAppPane } from '../Panes/ResponsivePaneProvider'
|
|
import { AppPaneId } from '../Panes/AppPaneMetadata'
|
|
import MenuItem from '../Menu/MenuItem'
|
|
import { FileContextMenuBackupOption } from './FileContextMenuBackupOption'
|
|
import MenuSwitchButtonItem from '../Menu/MenuSwitchButtonItem'
|
|
import { FileItem } from '@standardnotes/snjs'
|
|
import AddTagOption from '../NotesOptions/AddTagOption'
|
|
import { MenuItemIconSize } from '@/Constants/TailwindClassNames'
|
|
import AddToVaultMenuOption from '../Vaults/AddToVaultMenuOption'
|
|
import { iconClass } from '../NotesOptions/ClassNames'
|
|
import { useApplication } from '../ApplicationProvider'
|
|
import MenuSection from '../Menu/MenuSection'
|
|
import { ToastType, addToast } from '@standardnotes/toast'
|
|
|
|
type Props = {
|
|
closeMenu: () => void
|
|
isFileAttachedToNote?: boolean
|
|
renameToggleCallback?: (isRenamingFile: boolean) => void
|
|
shouldShowRenameOption: boolean
|
|
shouldShowAttachOption: boolean
|
|
selectedFiles: FileItem[]
|
|
}
|
|
|
|
const FileMenuOptions: FunctionComponent<Props> = ({
|
|
closeMenu,
|
|
isFileAttachedToNote,
|
|
renameToggleCallback,
|
|
shouldShowRenameOption,
|
|
shouldShowAttachOption,
|
|
selectedFiles,
|
|
}) => {
|
|
const application = useApplication()
|
|
|
|
const { shouldUseStreamingAPI, handleFileAction } = application.filesController
|
|
const { toggleAppPane } = useResponsiveAppPane()
|
|
|
|
const hasProtectedFiles = useMemo(() => selectedFiles.some((file) => file.protected), [selectedFiles])
|
|
const hasSelectedMultipleFiles = useMemo(() => selectedFiles.length > 1, [selectedFiles.length])
|
|
const canShowZipDownloadOption = shouldUseStreamingAPI && hasSelectedMultipleFiles
|
|
|
|
const totalFileSize = useMemo(
|
|
() => selectedFiles.map((file) => file.decryptedSize).reduce((prev, next) => prev + next, 0),
|
|
[selectedFiles],
|
|
)
|
|
|
|
const onDetach = useCallback(() => {
|
|
const file = selectedFiles[0]
|
|
void handleFileAction({
|
|
type: FileItemActionType.DetachFileToNote,
|
|
payload: { file },
|
|
})
|
|
closeMenu()
|
|
}, [closeMenu, handleFileAction, selectedFiles])
|
|
|
|
const onAttach = useCallback(() => {
|
|
const file = selectedFiles[0]
|
|
void handleFileAction({
|
|
type: FileItemActionType.AttachFileToNote,
|
|
payload: { file },
|
|
})
|
|
closeMenu()
|
|
}, [closeMenu, handleFileAction, selectedFiles])
|
|
|
|
const closeMenuAndToggleFilesList = useCallback(() => {
|
|
toggleAppPane(AppPaneId.Items)
|
|
closeMenu()
|
|
}, [closeMenu, toggleAppPane])
|
|
|
|
const areSomeFilesInReadonlySharedVault = selectedFiles.some((file) => {
|
|
const vault = application.vaults.getItemVault(file)
|
|
return vault?.isSharedVaultListing() && application.vaultUsers.isCurrentUserReadonlyVaultMember(vault)
|
|
})
|
|
const hasAdminPermissionForAllSharedFiles = selectedFiles.every((file) => {
|
|
const vault = application.vaults.getItemVault(file)
|
|
if (!vault?.isSharedVaultListing()) {
|
|
return true
|
|
}
|
|
return application.vaultUsers.isCurrentUserSharedVaultAdmin(vault)
|
|
})
|
|
|
|
if (selectedFiles.length === 0) {
|
|
return <div className="text-center">No files selected</div>
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{selectedFiles.length === 1 && (isFileAttachedToNote || shouldShowAttachOption) && (
|
|
<MenuSection>
|
|
{isFileAttachedToNote ? (
|
|
<MenuItem onClick={onDetach}>
|
|
<Icon type="link-off" className="mr-2 text-neutral" />
|
|
Detach from note
|
|
</MenuItem>
|
|
) : shouldShowAttachOption ? (
|
|
<MenuItem onClick={onAttach}>
|
|
<Icon type="link" className="mr-2 text-neutral" />
|
|
Attach to note
|
|
</MenuItem>
|
|
) : null}
|
|
</MenuSection>
|
|
)}
|
|
<MenuSection>
|
|
{application.featuresController.isVaultsEnabled() && (
|
|
<AddToVaultMenuOption
|
|
iconClassName={iconClass}
|
|
items={selectedFiles}
|
|
disabled={!hasAdminPermissionForAllSharedFiles}
|
|
/>
|
|
)}
|
|
<AddTagOption
|
|
navigationController={application.navigationController}
|
|
linkingController={application.linkingController}
|
|
selectedItems={selectedFiles}
|
|
iconClassName={`text-neutral mr-2 ${MenuItemIconSize}`}
|
|
disabled={areSomeFilesInReadonlySharedVault}
|
|
/>
|
|
<MenuSwitchButtonItem
|
|
checked={hasProtectedFiles}
|
|
onChange={(hasProtectedFiles) => {
|
|
void application.filesController.setProtectionForFiles(hasProtectedFiles, selectedFiles)
|
|
}}
|
|
disabled={areSomeFilesInReadonlySharedVault}
|
|
>
|
|
<Icon type="lock" className={`mr-2 text-neutral ${MenuItemIconSize}`} />
|
|
Password protect
|
|
</MenuSwitchButtonItem>
|
|
</MenuSection>
|
|
<MenuSection>
|
|
<MenuItem
|
|
onClick={() => {
|
|
void application.filesController.downloadFiles(selectedFiles)
|
|
closeMenu()
|
|
}}
|
|
>
|
|
<Icon type="download" className={`mr-2 text-neutral ${MenuItemIconSize}`} />
|
|
Download {canShowZipDownloadOption ? 'separately' : ''}
|
|
</MenuItem>
|
|
{canShowZipDownloadOption && (
|
|
<MenuItem
|
|
onClick={() => {
|
|
application.filesController.downloadFilesAsZip(selectedFiles).catch((error) => {
|
|
if (error instanceof DOMException && error.name === 'AbortError') {
|
|
return
|
|
}
|
|
console.error(error)
|
|
addToast({
|
|
type: ToastType.Error,
|
|
message: error.message || 'Failed to download files as archive',
|
|
})
|
|
})
|
|
closeMenu()
|
|
}}
|
|
>
|
|
<Icon type="download" className={`mr-2 text-neutral ${MenuItemIconSize}`} />
|
|
Download as archive
|
|
</MenuItem>
|
|
)}
|
|
{shouldShowRenameOption && (
|
|
<MenuItem
|
|
onClick={() => {
|
|
renameToggleCallback?.(true)
|
|
}}
|
|
disabled={areSomeFilesInReadonlySharedVault}
|
|
>
|
|
<Icon type="pencil" className={`mr-2 text-neutral ${MenuItemIconSize}`} />
|
|
Rename
|
|
</MenuItem>
|
|
)}
|
|
<MenuItem
|
|
onClick={() => {
|
|
closeMenuAndToggleFilesList()
|
|
void application.filesController.deleteFilesPermanently(selectedFiles)
|
|
}}
|
|
disabled={areSomeFilesInReadonlySharedVault}
|
|
>
|
|
<Icon type="trash" className={`mr-2 text-danger ${MenuItemIconSize}`} />
|
|
<span className="text-danger">Delete permanently</span>
|
|
</MenuItem>
|
|
</MenuSection>
|
|
|
|
<FileContextMenuBackupOption file={selectedFiles[0]} />
|
|
|
|
<div className="px-3 pb-0.5 pt-1 text-xs font-medium text-neutral">
|
|
{!hasSelectedMultipleFiles && (
|
|
<div className="mb-1">
|
|
<span className="font-semibold">File ID:</span> {selectedFiles[0].uuid}
|
|
</div>
|
|
)}
|
|
<div>
|
|
<span className="font-semibold">{hasSelectedMultipleFiles ? 'Total Size:' : 'Size:'}</span>{' '}
|
|
{formatSizeToReadableString(totalFileSize)}
|
|
</div>
|
|
</div>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default observer(FileMenuOptions)
|