"use client"; import React, { useRef, useState } from "react"; import { observer } from "mobx-react"; import Link from "next/link"; import { useParams } from "next/navigation"; import { ArchiveRestoreIcon, Check, ExternalLink, LinkIcon, Lock, Settings, Trash2, UserPlus } from "lucide-react"; // plane imports import { EUserPermissions, EUserPermissionsLevel, IS_FAVORITE_MENU_OPEN } from "@plane/constants"; import { useLocalStorage } from "@plane/hooks"; import { Button } from "@plane/propel/button"; import { setPromiseToast, setToast, TOAST_TYPE } from "@plane/propel/toast"; import { Tooltip } from "@plane/propel/tooltip"; import type { IProject } from "@plane/types"; import type { TContextMenuItem } from "@plane/ui"; import { Avatar, AvatarGroup, ContextMenu, FavoriteStar } from "@plane/ui"; import { copyUrlToClipboard, cn, getFileURL, renderFormattedDate } from "@plane/utils"; // components import { Logo } from "@/components/common/logo"; // hooks import { useMember } from "@/hooks/store/use-member"; import { useProject } from "@/hooks/store/use-project"; import { useUserPermissions } from "@/hooks/store/user"; import { useAppRouter } from "@/hooks/use-app-router"; import { usePlatformOS } from "@/hooks/use-platform-os"; // local imports import { DeleteProjectModal } from "./delete-project-modal"; import { JoinProjectModal } from "./join-project-modal"; import { ArchiveRestoreProjectModal } from "./settings/archive-project/archive-restore-modal"; type Props = { project: IProject; }; export const ProjectCard: React.FC = observer((props) => { const { project } = props; // states const [deleteProjectModalOpen, setDeleteProjectModal] = useState(false); const [joinProjectModalOpen, setJoinProjectModal] = useState(false); const [restoreProject, setRestoreProject] = useState(false); // refs const projectCardRef = useRef(null); // router const router = useAppRouter(); const { workspaceSlug } = useParams(); // store hooks const { getUserDetails } = useMember(); const { addProjectToFavorites, removeProjectFromFavorites } = useProject(); const { allowPermissions } = useUserPermissions(); // hooks const { isMobile } = usePlatformOS(); // derived values const projectMembersIds = project.members; const shouldRenderFavorite = allowPermissions( [EUserPermissions.ADMIN, EUserPermissions.MEMBER], EUserPermissionsLevel.WORKSPACE ); // auth const isMemberOfProject = !!project.member_role; const hasAdminRole = project.member_role === EUserPermissions.ADMIN; const hasMemberRole = project.member_role === EUserPermissions.MEMBER; // archive const isArchived = !!project.archived_at; // local storage const { setValue: toggleFavoriteMenu, storedValue: isFavoriteMenuOpen } = useLocalStorage( IS_FAVORITE_MENU_OPEN, false ); const handleAddToFavorites = () => { if (!workspaceSlug) return; const addToFavoritePromise = addProjectToFavorites(workspaceSlug.toString(), project.id); setPromiseToast(addToFavoritePromise, { loading: "Adding project to favorites...", success: { title: "Success!", message: () => "Project added to favorites.", actionItems: () => { if (!isFavoriteMenuOpen) toggleFavoriteMenu(true); return <>; }, }, error: { title: "Error!", message: () => "Couldn't add the project to favorites. Please try again.", }, }); }; const handleRemoveFromFavorites = () => { if (!workspaceSlug) return; const removeFromFavoritePromise = removeProjectFromFavorites(workspaceSlug.toString(), project.id); setPromiseToast(removeFromFavoritePromise, { loading: "Removing project from favorites...", success: { title: "Success!", message: () => "Project removed from favorites.", }, error: { title: "Error!", message: () => "Couldn't remove the project from favorites. Please try again.", }, }); }; const projectLink = `${workspaceSlug}/projects/${project.id}/issues`; const handleCopyText = () => copyUrlToClipboard(projectLink).then(() => setToast({ type: TOAST_TYPE.INFO, title: "Link Copied!", message: "Project link copied to clipboard.", }) ); const handleOpenInNewTab = () => window.open(`/${projectLink}`, "_blank"); const MENU_ITEMS: TContextMenuItem[] = [ { key: "settings", action: () => router.push(`/${workspaceSlug}/settings/projects/${project.id}`, { showProgress: false }), title: "Settings", icon: Settings, shouldRender: !isArchived && (hasAdminRole || hasMemberRole), }, { key: "join", action: () => setJoinProjectModal(true), title: "Join", icon: UserPlus, shouldRender: !isMemberOfProject && !isArchived, }, { key: "open-new-tab", action: handleOpenInNewTab, title: "Open in new tab", icon: ExternalLink, shouldRender: !isMemberOfProject && !isArchived, }, { key: "copy-link", action: handleCopyText, title: "Copy link", icon: LinkIcon, shouldRender: !isArchived, }, { key: "restore", action: () => setRestoreProject(true), title: "Restore", icon: ArchiveRestoreIcon, shouldRender: isArchived && hasAdminRole, }, { key: "delete", action: () => setDeleteProjectModal(true), title: "Delete", icon: Trash2, shouldRender: isArchived && hasAdminRole, }, ]; return ( <> {/* Delete Project Modal */} setDeleteProjectModal(false)} /> {/* Join Project Modal */} {workspaceSlug && ( setJoinProjectModal(false)} /> )} {/* Restore project modal */} {workspaceSlug && project && ( setRestoreProject(false)} archive={false} /> )} { if (!isMemberOfProject || isArchived) { e.preventDefault(); e.stopPropagation(); if (!isArchived) setJoinProjectModal(true); } }} data-prevent-progress={!isMemberOfProject || isArchived} className="flex flex-col rounded border border-custom-border-200 bg-custom-background-100" >
{project.name}

{project.name}

{project.identifier}

{project.network === 0 && }
{!isArchived && (
{shouldRenderFavorite && ( { e.preventDefault(); e.stopPropagation(); if (project.is_favorite) handleRemoveFromFavorites(); else handleAddToFavorites(); }} selected={!!project.is_favorite} /> )}
)}

{project.description && project.description.trim() !== "" ? project.description : `Created on ${renderFormattedDate(project.created_at)}`}

0 ? `${project.members.length} Members` : "No Member" } position="top" > {projectMembersIds && projectMembersIds.length > 0 ? (
{projectMembersIds.map((memberId) => { const member = getUserDetails(memberId); if (!member) return null; return ( ); })}
) : ( No Member Yet )}
{isArchived &&
Archived
}
{isArchived ? ( hasAdminRole && (
{ e.preventDefault(); e.stopPropagation(); setRestoreProject(true); }} >
Restore
{ e.preventDefault(); e.stopPropagation(); setDeleteProjectModal(true); }} >
) ) : ( <> {isMemberOfProject && (hasAdminRole || hasMemberRole ? ( { e.stopPropagation(); }} href={`/${workspaceSlug}/settings/projects/${project.id}`} > ) : ( Joined ))} {!isMemberOfProject && (
)} )}
); });