import React, { useCallback, useEffect, FC } from "react"; import { observer } from "mobx-react-lite"; import { useRouter } from "next/router"; import useSWR from "swr"; // hooks import { TOAST_TYPE, setToast } from "@plane/ui"; import { CommandModal, ShortcutsModal } from "components/command-palette"; // ui // components import { BulkDeleteIssuesModal } from "components/core"; import { CycleCreateUpdateModal } from "components/cycles"; import { CreateUpdateIssueModal, DeleteIssueModal } from "components/issues"; import { CreateUpdateModuleModal } from "components/modules"; import { CreateUpdatePageModal } from "components/pages"; import { CreateProjectModal } from "components/project"; import { CreateUpdateProjectViewModal } from "components/views"; // helpers // services // fetch keys import { ISSUE_DETAILS } from "constants/fetch-keys"; import { EIssuesStoreType } from "constants/issue"; import { copyTextToClipboard } from "helpers/string.helper"; import { useApplication, useEventTracker, useIssues, useUser } from "hooks/store"; import { IssueService } from "services/issue"; // constants import { E_SHORTCUT_KEY } from "constants/event-tracker"; // services const issueService = new IssueService(); export const CommandPalette: FC = observer(() => { const router = useRouter(); const { workspaceSlug, projectId, issueId, cycleId, moduleId } = router.query; const { commandPalette, theme: { toggleSidebar }, } = useApplication(); const { setTrackElement } = useEventTracker(); const { currentUser } = useUser(); const { issues: { removeIssue }, } = useIssues(EIssuesStoreType.PROJECT); const { toggleCommandPaletteModal, isCreateIssueModalOpen, toggleCreateIssueModal, isCreateCycleModalOpen, toggleCreateCycleModal, isCreatePageModalOpen, toggleCreatePageModal, isCreateProjectModalOpen, toggleCreateProjectModal, isCreateModuleModalOpen, toggleCreateModuleModal, isCreateViewModalOpen, toggleCreateViewModal, isShortcutModalOpen, toggleShortcutModal, isBulkDeleteIssueModalOpen, toggleBulkDeleteIssueModal, isDeleteIssueModalOpen, toggleDeleteIssueModal, isAnyModalOpen, createIssueStoreType, } = commandPalette; const { data: issueDetails } = useSWR( workspaceSlug && projectId && issueId ? ISSUE_DETAILS(issueId as string) : null, workspaceSlug && projectId && issueId ? () => issueService.retrieve(workspaceSlug as string, projectId as string, issueId as string) : null ); const copyIssueUrlToClipboard = useCallback(() => { if (!issueId) return; const url = new URL(window.location.href); copyTextToClipboard(url.href) .then(() => { setToast({ type: TOAST_TYPE.SUCCESS, title: "Copied to clipboard", }); }) .catch(() => { setToast({ type: TOAST_TYPE.ERROR, title: "Some error occurred", }); }); }, [issueId]); const handleKeyDown = useCallback( (e: KeyboardEvent) => { const { key, ctrlKey, metaKey, altKey } = e; if (!key) return; const keyPressed = key.toLowerCase(); const cmdClicked = ctrlKey || metaKey; // if on input, textarea or editor, don't do anything if ( e.target instanceof HTMLTextAreaElement || e.target instanceof HTMLInputElement || (e.target as Element).classList?.contains("ProseMirror") ) return; if (cmdClicked) { if (keyPressed === "k") { e.preventDefault(); toggleCommandPaletteModal(true); } else if (keyPressed === "c" && altKey) { e.preventDefault(); copyIssueUrlToClipboard(); } else if (keyPressed === "b") { e.preventDefault(); toggleSidebar(); } } else if (!isAnyModalOpen) { setTrackElement(E_SHORTCUT_KEY); if (keyPressed === "c") { toggleCreateIssueModal(true); } else if (keyPressed === "p") { toggleCreateProjectModal(true); } else if (keyPressed === "h") { toggleShortcutModal(true); } else if (keyPressed === "v" && workspaceSlug && projectId) { toggleCreateViewModal(true); } else if (keyPressed === "d" && workspaceSlug && projectId) { toggleCreatePageModal(true); } else if (keyPressed === "q" && workspaceSlug && projectId) { toggleCreateCycleModal(true); } else if (keyPressed === "m" && workspaceSlug && projectId) { toggleCreateModuleModal(true); } else if (keyPressed === "backspace" || keyPressed === "delete") { e.preventDefault(); toggleBulkDeleteIssueModal(true); } } }, [ copyIssueUrlToClipboard, toggleCreateProjectModal, toggleCreateViewModal, toggleCreatePageModal, toggleShortcutModal, toggleCreateCycleModal, toggleCreateModuleModal, toggleBulkDeleteIssueModal, toggleCommandPaletteModal, toggleSidebar, toggleCreateIssueModal, projectId, workspaceSlug, isAnyModalOpen, setTrackElement, ] ); useEffect(() => { document.addEventListener("keydown", handleKeyDown); return () => document.removeEventListener("keydown", handleKeyDown); }, [handleKeyDown]); const isDraftIssue = router?.asPath?.includes("draft-issues") || false; if (!currentUser) return null; return ( <> { toggleShortcutModal(false); }} /> {workspaceSlug && ( { toggleCreateProjectModal(false); }} workspaceSlug={workspaceSlug.toString()} /> )} {workspaceSlug && projectId && ( <> toggleCreateCycleModal(false)} workspaceSlug={workspaceSlug.toString()} projectId={projectId.toString()} /> { toggleCreateModuleModal(false); }} workspaceSlug={workspaceSlug.toString()} projectId={projectId.toString()} /> toggleCreateViewModal(false)} workspaceSlug={workspaceSlug.toString()} projectId={projectId.toString()} /> toggleCreatePageModal(false)} projectId={projectId.toString()} /> )} toggleCreateIssueModal(false)} data={cycleId ? { cycle_id: cycleId.toString() } : moduleId ? { module_ids: [moduleId.toString()] } : undefined} storeType={createIssueStoreType} isDraft={isDraftIssue} /> {workspaceSlug && projectId && issueId && issueDetails && ( toggleDeleteIssueModal(false)} isOpen={isDeleteIssueModalOpen} data={issueDetails} onSubmit={async () => { await removeIssue(workspaceSlug.toString(), projectId.toString(), issueId.toString()); router.push(`/${workspaceSlug}/projects/${projectId}/issues`); }} /> )} { toggleBulkDeleteIssueModal(false); }} user={currentUser} /> ); });