import { FC, useState } from "react"; import { observer } from "mobx-react-lite"; import Link from "next/link"; import { useRouter } from "next/router"; import { AlertCircle, Archive, ArchiveRestoreIcon, FileText, Globe2, LinkIcon, Lock, Pencil, Star, Trash2, } from "lucide-react"; // ui import { CustomMenu, Tooltip } from "@plane/ui"; // components import { CreateUpdatePageModal, DeletePageModal } from "components/pages"; // constants import { EUserProjectRoles } from "constants/project"; import { E_PROJECT_PAGES, PAGE_ARCHIVED, PAGE_FAVORITED, PAGE_RESTORED, PAGE_UNFAVORITED, PAGE_UPDATED, PAGE_VIEWED, } from "constants/event-tracker"; import { renderFormattedTime, renderFormattedDate } from "helpers/date-time.helper"; import { copyUrlToClipboard } from "helpers/string.helper"; // hooks import { useEventTracker, useMember, usePage, useUser } from "hooks/store"; import { useProjectPages } from "hooks/store/use-project-specific-pages"; import { usePlatformOS } from "hooks/use-platform-os"; import { IIssueLabel } from "@plane/types"; export interface IPagesListItem { pageId: string; projectId: string; } export const PagesListItem: FC = observer(({ pageId, projectId }: IPagesListItem) => { const projectPageStore = useProjectPages(); // Now, I am observing only the projectPages, out of the projectPageStore. const { archivePage, restorePage } = projectPageStore; const pageStore = usePage(pageId); // router const router = useRouter(); const { workspaceSlug } = router.query; // states const [createUpdatePageModal, setCreateUpdatePageModal] = useState(false); const [deletePageModal, setDeletePageModal] = useState(false); // store hooks const { isMobile } = usePlatformOS(); const { currentUser, membership: { currentProjectRole }, } = useUser(); const { project: { getProjectMemberDetails }, } = useMember(); const { captureEvent, capturePageEvent } = useEventTracker(); if (!pageStore) return null; const { archived_at, label_details, access, is_favorite, owned_by, name, created_at, updated_at, makePublic, makePrivate, addToFavorites, removeFromFavorites, } = pageStore; const handleCopyUrl = async (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); await copyUrlToClipboard(`${workspaceSlug}/projects/${projectId}/pages/${pageId}`); }; const handleAddToFavorites = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); addToFavorites().then(() => { captureEvent(PAGE_FAVORITED, { page_id: pageId, element: E_PROJECT_PAGES, state: "SUCCESS", }); }); }; const handleRemoveFromFavorites = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); removeFromFavorites().then(() => { captureEvent(PAGE_UNFAVORITED, { page_id: pageId, element: E_PROJECT_PAGES, state: "SUCCESS", }); }); }; const handleMakePublic = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); makePublic().then(() => { captureEvent(PAGE_UPDATED, { page_id: pageId, access: "public", element: E_PROJECT_PAGES, state: "SUCCESS", }); }); }; const handleMakePrivate = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); makePrivate().then(() => { captureEvent(PAGE_UPDATED, { page_id: pageId, access: "private", element: E_PROJECT_PAGES, state: "SUCCESS", }); }); }; const handleArchivePage = async (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); await archivePage(workspaceSlug as string, projectId as string, pageId as string).then(() => { captureEvent(PAGE_ARCHIVED, { page_id: pageId, access: access == 1 ? "private" : "public", element: E_PROJECT_PAGES, state: "SUCCESS", }); }); }; const handleRestorePage = async (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); await restorePage(workspaceSlug as string, projectId as string, pageId as string).then(() => { captureEvent(PAGE_RESTORED, { page_id: pageId, access: access == 1 ? "private" : "public", element: E_PROJECT_PAGES, state: "SUCCESS", }); }); }; const handleDeletePage = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); setDeletePageModal(true); }; const handleEditPage = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); setCreateUpdatePageModal(true); }; const ownerDetails = getProjectMemberDetails(owned_by); const isCurrentUserOwner = owned_by === currentUser?.id; const userCanEdit = isCurrentUserOwner || (currentProjectRole && [EUserProjectRoles.ADMIN, EUserProjectRoles.MEMBER].includes(currentProjectRole)); const userCanChangeAccess = isCurrentUserOwner; const userCanArchive = isCurrentUserOwner || currentProjectRole === EUserProjectRoles.ADMIN; const userCanDelete = isCurrentUserOwner || currentProjectRole === EUserProjectRoles.ADMIN; const isEditingAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER; return ( <> setCreateUpdatePageModal(false)} projectId={projectId} /> setDeletePageModal(false)} pageId={pageId} />
  • capturePageEvent({ eventName: PAGE_VIEWED, payload: { ...pageStore, element: E_PROJECT_PAGES, }, }) } >

    {name}

    {label_details.length > 0 && label_details.map((label: IIssueLabel) => (
    {label.name}
    ))}
    {archived_at ? (

    {renderFormattedTime(archived_at)}

    ) : (

    {renderFormattedTime(updated_at)}

    )} {isEditingAllowed && ( {is_favorite ? ( ) : ( )} )} {userCanChangeAccess && ( {access ? ( ) : ( )} )} {archived_at ? ( <> {userCanArchive && (
    Restore page
    )} {userCanDelete && isEditingAllowed && (
    Delete page
    )} ) : ( <> {userCanEdit && isEditingAllowed && (
    Edit page
    )} {userCanArchive && isEditingAllowed && (
    Archive page
    )} )}
    Copy page link
  • ); });