"use client"; import { useState } from "react"; import { observer } from "mobx-react"; // icons import { ArchiveRestoreIcon, ExternalLink, LinkIcon, Pencil, Trash2 } from "lucide-react"; // plane imports import { EUserPermissions, EUserPermissionsLevel, MODULE_TRACKER_ELEMENTS, MODULE_TRACKER_EVENTS, } from "@plane/constants"; import { useTranslation } from "@plane/i18n"; // ui import { ArchiveIcon } from "@plane/propel/icons"; import { ContextMenu, CustomMenu, TContextMenuItem, TOAST_TYPE, setToast } from "@plane/ui"; import { copyUrlToClipboard, cn } from "@plane/utils"; // components import { ArchiveModuleModal, CreateUpdateModuleModal, DeleteModuleModal } from "@/components/modules"; // helpers import { captureClick, captureSuccess, captureError } from "@/helpers/event-tracker.helper"; // hooks import { useModule } from "@/hooks/store/use-module"; import { useUserPermissions } from "@/hooks/store/user"; import { useAppRouter } from "@/hooks/use-app-router"; type Props = { parentRef: React.RefObject; moduleId: string; projectId: string; workspaceSlug: string; customClassName?: string; }; export const ModuleQuickActions: React.FC = observer((props) => { const { parentRef, moduleId, projectId, workspaceSlug, customClassName } = props; // router const router = useAppRouter(); // states const [editModal, setEditModal] = useState(false); const [archiveModuleModal, setArchiveModuleModal] = useState(false); const [deleteModal, setDeleteModal] = useState(false); // store hooks const { allowPermissions } = useUserPermissions(); const { getModuleById, restoreModule } = useModule(); const { t } = useTranslation(); // derived values const moduleDetails = getModuleById(moduleId); const isArchived = !!moduleDetails?.archived_at; // auth const isEditingAllowed = allowPermissions( [EUserPermissions.ADMIN, EUserPermissions.MEMBER], EUserPermissionsLevel.PROJECT, workspaceSlug, projectId ); const moduleState = moduleDetails?.status?.toLocaleLowerCase(); const isInArchivableGroup = !!moduleState && ["completed", "cancelled"].includes(moduleState); const moduleLink = `${workspaceSlug}/projects/${projectId}/modules/${moduleId}`; const handleCopyText = () => copyUrlToClipboard(moduleLink).then(() => { setToast({ type: TOAST_TYPE.SUCCESS, title: "Link Copied!", message: "Module link copied to clipboard.", }); }); const handleOpenInNewTab = () => window.open(`/${moduleLink}`, "_blank"); const handleEditModule = () => { setEditModal(true); }; const handleArchiveModule = () => setArchiveModuleModal(true); const handleRestoreModule = async () => await restoreModule(workspaceSlug, projectId, moduleId) .then(() => { setToast({ type: TOAST_TYPE.SUCCESS, title: "Restore success", message: "Your module can be found in project modules.", }); captureSuccess({ eventName: MODULE_TRACKER_EVENTS.restore, payload: { id: moduleId }, }); router.push(`/${workspaceSlug}/projects/${projectId}/archives/modules`); }) .catch((error) => { setToast({ type: TOAST_TYPE.ERROR, title: "Error!", message: "Module could not be restored. Please try again.", }); captureError({ eventName: MODULE_TRACKER_EVENTS.restore, payload: { id: moduleId }, error, }); }); const handleDeleteModule = () => { setDeleteModal(true); }; const MENU_ITEMS: TContextMenuItem[] = [ { key: "edit", title: t("edit"), icon: Pencil, action: handleEditModule, shouldRender: isEditingAllowed && !isArchived, }, { key: "open-new-tab", action: handleOpenInNewTab, title: t("open_in_new_tab"), icon: ExternalLink, shouldRender: !isArchived, }, { key: "copy-link", action: handleCopyText, title: t("copy_link"), icon: LinkIcon, shouldRender: !isArchived, }, { key: "archive", action: handleArchiveModule, title: t("archive"), description: isInArchivableGroup ? undefined : t("project_module.quick_actions.archive_module_description"), icon: ArchiveIcon, className: "items-start", iconClassName: "mt-1", shouldRender: isEditingAllowed && !isArchived, disabled: !isInArchivableGroup, }, { key: "restore", action: handleRestoreModule, title: t("restore"), icon: ArchiveRestoreIcon, shouldRender: isEditingAllowed && isArchived, }, { key: "delete", action: handleDeleteModule, title: t("delete"), icon: Trash2, shouldRender: isEditingAllowed, }, ]; const CONTEXT_MENU_ITEMS: TContextMenuItem[] = MENU_ITEMS.map((item) => ({ ...item, onClick: () => { captureClick({ elementName: MODULE_TRACKER_ELEMENTS.CONTEXT_MENU, }); item.action(); }, })); return ( <> {moduleDetails && (
setEditModal(false)} data={moduleDetails} projectId={projectId} workspaceSlug={workspaceSlug} /> setArchiveModuleModal(false)} /> setDeleteModal(false)} />
)} {MENU_ITEMS.map((item) => { if (item.shouldRender === false) return null; return ( { e.preventDefault(); e.stopPropagation(); captureClick({ elementName: MODULE_TRACKER_ELEMENTS.QUICK_ACTIONS, }); item.action(); }} className={cn( "flex items-center gap-2", { "text-custom-text-400": item.disabled, }, item.className )} disabled={item.disabled} > {item.icon && }
{item.title}
{item.description && (

{item.description}

)}
); })}
); });