import { Fragment, useState, ReactElement } from "react"; import { observer } from "mobx-react-lite"; import { useRouter } from "next/router"; import { Tab } from "@headlessui/react"; // hooks import { useEventTracker, useCycle, useProject, useCycleFilter } from "hooks/store"; // layouts import { AppLayout } from "layouts/app-layout"; // components import { PageHead } from "components/core"; import { CyclesHeader } from "components/headers"; import { CyclesView, CycleCreateUpdateModal, CyclesViewHeader, CycleAppliedFiltersList, ActiveCycleRoot, } from "components/cycles"; import { EmptyState } from "components/empty-state"; // ui import { CycleModuleBoardLayout, CycleModuleListLayout, GanttLayoutLoader } from "components/ui"; // helpers import { calculateTotalFilters } from "helpers/filter.helper"; // types import { NextPageWithLayout } from "lib/types"; import { TCycleFilters } from "@plane/types"; // constants import { CYCLE_TABS_LIST } from "constants/cycle"; import { EmptyStateType } from "constants/empty-state"; import { E_CYCLES_EMPTY_STATE } from "constants/event-tracker"; import CyclesListMobileHeader from "components/cycles/cycles-list-mobile-header"; const ProjectCyclesPage: NextPageWithLayout = observer(() => { // states const [createModal, setCreateModal] = useState(false); // store hooks const { setTrackElement } = useEventTracker(); const { currentProjectCycleIds, loader } = useCycle(); const { getProjectById } = useProject(); // router const router = useRouter(); const { workspaceSlug, projectId, peekCycle } = router.query; // cycle filters hook const { clearAllFilters, currentProjectDisplayFilters, currentProjectFilters, updateDisplayFilters, updateFilters } = useCycleFilter(); // derived values const totalCycles = currentProjectCycleIds?.length ?? 0; const project = projectId ? getProjectById(projectId?.toString()) : undefined; const pageTitle = project?.name ? `${project?.name} - Cycles` : undefined; // selected display filters const cycleTab = currentProjectDisplayFilters?.active_tab; const cycleLayout = currentProjectDisplayFilters?.layout ?? "list"; const handleRemoveFilter = (key: keyof TCycleFilters, value: string | null) => { if (!projectId) return; let newValues = currentProjectFilters?.[key] ?? []; if (!value) newValues = []; else newValues = newValues.filter((val) => val !== value); updateFilters(projectId.toString(), { [key]: newValues }); }; if (!workspaceSlug || !projectId) return null; if (loader) return ( <> {cycleLayout === "list" && } {cycleLayout === "board" && } {cycleLayout === "gantt" && } ); return ( <>
setCreateModal(false)} /> {totalCycles === 0 ? (
{ setTrackElement(E_CYCLES_EMPTY_STATE); setCreateModal(true); }} />
) : ( i.key == cycleTab)} selectedIndex={CYCLE_TABS_LIST.findIndex((i) => i.key == cycleTab)} onChange={(i) => { if (!projectId) return; const tab = CYCLE_TABS_LIST[i]; if (!tab) return; updateDisplayFilters(projectId.toString(), { active_tab: tab.key, }); }} > {calculateTotalFilters(currentProjectFilters ?? {}) !== 0 && (
clearAllFilters(projectId.toString())} handleRemoveFilter={handleRemoveFilter} />
)}
)}
); }); ProjectCyclesPage.getLayout = function getLayout(page: ReactElement) { return ( } mobileHeader={} withProjectWrapper> {page} ); }; export default ProjectCyclesPage;