import { Dispatch, MouseEvent, MutableRefObject, SetStateAction, useRef, useState } from "react"; import { observer } from "mobx-react"; import { useRouter } from "next/router"; // icons import { ChevronRight, MoreHorizontal } from "lucide-react"; import { IIssueDisplayProperties, TIssue } from "@plane/types"; // ui import { ControlLink, Tooltip } from "@plane/ui"; // components import RenderIfVisible from "@/components/core/render-if-visible-HOC"; // helper import { cn } from "@/helpers/common.helper"; // hooks import { useIssueDetail, useProject } from "@/hooks/store"; import useOutsideClickDetector from "@/hooks/use-outside-click-detector"; import { usePlatformOS } from "@/hooks/use-platform-os"; // types // local components import { TRenderQuickActions } from "../list/list-view-types"; import { WithDisplayPropertiesHOC } from "../properties/with-display-properties-HOC"; import { IssueColumn } from "./issue-column"; interface Props { displayProperties: IIssueDisplayProperties; isEstimateEnabled: boolean; quickActions: TRenderQuickActions; canEditProperties: (projectId: string | undefined) => boolean; updateIssue: ((projectId: string, issueId: string, data: Partial) => Promise) | undefined; portalElement: React.MutableRefObject; nestingLevel: number; issueId: string; isScrolled: MutableRefObject; containerRef: MutableRefObject; issueIds: string[]; spreadsheetColumnsList: (keyof IIssueDisplayProperties)[]; spacingLeft?: number; } export const SpreadsheetIssueRow = observer((props: Props) => { const { displayProperties, issueId, isEstimateEnabled, nestingLevel, portalElement, updateIssue, quickActions, canEditProperties, isScrolled, containerRef, issueIds, spreadsheetColumnsList, spacingLeft = 6, } = props; const [isExpanded, setExpanded] = useState(false); const { subIssues: subIssuesStore } = useIssueDetail(); const subIssues = subIssuesStore.subIssuesByIssueId(issueId); return ( <> {/* first column/ issue name and key column */} } > {isExpanded && subIssues && subIssues.length > 0 && subIssues.map((subIssueId: string) => ( ))} ); }); interface IssueRowDetailsProps { displayProperties: IIssueDisplayProperties; isEstimateEnabled: boolean; quickActions: TRenderQuickActions; canEditProperties: (projectId: string | undefined) => boolean; updateIssue: ((projectId: string, issueId: string, data: Partial) => Promise) | undefined; portalElement: React.MutableRefObject; nestingLevel: number; issueId: string; isScrolled: MutableRefObject; isExpanded: boolean; setExpanded: Dispatch>; spreadsheetColumnsList: (keyof IIssueDisplayProperties)[]; spacingLeft?: number; } const IssueRowDetails = observer((props: IssueRowDetailsProps) => { const { displayProperties, issueId, isEstimateEnabled, nestingLevel, portalElement, updateIssue, quickActions, canEditProperties, isScrolled, isExpanded, setExpanded, spreadsheetColumnsList, spacingLeft = 6, } = props; // states const [isMenuActive, setIsMenuActive] = useState(false); // refs const cellRef = useRef(null); const menuActionRef = useRef(null); // router const router = useRouter(); const { workspaceSlug } = router.query; // hooks const { getProjectIdentifierById } = useProject(); const { getIsIssuePeeked, setPeekIssue } = useIssueDetail(); const { isMobile } = usePlatformOS(); const handleIssuePeekOverview = (issue: TIssue) => workspaceSlug && issue && issue.project_id && issue.id && !getIsIssuePeeked(issue.id) && setPeekIssue({ workspaceSlug: workspaceSlug.toString(), projectId: issue.project_id, issueId: issue.id }); const { subIssues: subIssuesStore, issue } = useIssueDetail(); const issueDetail = issue.getIssueById(issueId); const subIssues = subIssuesStore.subIssuesByIssueId(issueId); const paddingLeft = `${spacingLeft}px`; useOutsideClickDetector(menuActionRef, () => setIsMenuActive(false)); const customActionButton = (
setIsMenuActive(!isMenuActive)} >
); if (!issueDetail) return null; const handleToggleExpand = (e: MouseEvent) => { e.stopPropagation(); e.preventDefault(); if (nestingLevel >= 3) { handleIssuePeekOverview(issueDetail); } else { setExpanded((prevState) => { if (!prevState && workspaceSlug && issueDetail) subIssuesStore.fetchSubIssues(workspaceSlug.toString(), issueDetail.project_id, issueDetail.id); return !prevState; }); } }; const disableUserActions = !canEditProperties(issueDetail.project_id); // if sub issues have been fetched for the issue, use that for count or use issue's sub_issues_count const subIssuesCount = subIssues ? subIssues.length : issueDetail.sub_issues_count; return ( <> handleIssuePeekOverview(issueDetail)} className={cn( "group clickable cursor-pointer h-11 w-[28rem] flex items-center bg-custom-background-100 text-sm after:absolute border-r-[0.5px] z-10 border-custom-border-200", { "border-b-[0.5px]": !getIsIssuePeeked(issueDetail.id), "border border-custom-primary-70 hover:border-custom-primary-70": getIsIssuePeeked(issueDetail.id), "shadow-[8px_22px_22px_10px_rgba(0,0,0,0.05)]": isScrolled.current, } )} disabled={!!issueDetail?.tempId} >
{/* bulk ops */}
{subIssuesCount > 0 && ( )}

{getProjectIdentifierById(issueDetail.project_id)}-{issueDetail.sequence_id}

{issueDetail.name}
{quickActions({ issue: issueDetail, parentRef: cellRef, customActionButton, portalElement: portalElement.current, })}
{/* Rest of the columns */} {spreadsheetColumnsList.map((property) => ( ))} ); });