mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
82 lines
3.0 KiB
TypeScript
82 lines
3.0 KiB
TypeScript
import { observer } from "mobx-react-lite";
|
||
import { useTheme } from "next-themes";
|
||
// hooks
|
||
import { useApplication, useEventTracker, useProject, useUser } from "hooks/store";
|
||
// components
|
||
import { ProjectCard } from "components/project";
|
||
import { Loader } from "@plane/ui";
|
||
import { EmptyState, getEmptyStateImagePath } from "components/empty-state";
|
||
// constants
|
||
import { EUserWorkspaceRoles } from "constants/workspace";
|
||
|
||
export const ProjectCardList = observer(() => {
|
||
// theme
|
||
const { resolvedTheme } = useTheme();
|
||
// store hooks
|
||
const { commandPalette: commandPaletteStore } = useApplication();
|
||
const { setTrackElement } = useEventTracker();
|
||
const {
|
||
membership: { currentWorkspaceRole },
|
||
currentUser,
|
||
} = useUser();
|
||
const { workspaceProjectIds, searchedProjects, getProjectById } = useProject();
|
||
|
||
const isLightMode = resolvedTheme ? resolvedTheme === "light" : currentUser?.theme.theme === "light";
|
||
const emptyStateImage = getEmptyStateImagePath("onboarding", "projects", isLightMode);
|
||
|
||
const isEditingAllowed = !!currentWorkspaceRole && currentWorkspaceRole >= EUserWorkspaceRoles.MEMBER;
|
||
|
||
if (!workspaceProjectIds)
|
||
return (
|
||
<Loader className="grid grid-cols-3 gap-4">
|
||
<Loader.Item height="100px" />
|
||
<Loader.Item height="100px" />
|
||
<Loader.Item height="100px" />
|
||
<Loader.Item height="100px" />
|
||
<Loader.Item height="100px" />
|
||
<Loader.Item height="100px" />
|
||
</Loader>
|
||
);
|
||
|
||
return (
|
||
<>
|
||
{workspaceProjectIds.length > 0 ? (
|
||
<div className="h-full w-full overflow-y-auto p-8">
|
||
{searchedProjects.length == 0 ? (
|
||
<div className="mt-10 w-full text-center text-neutral-text-subtle">No matching projects</div>
|
||
) : (
|
||
<div className="grid grid-cols-1 gap-9 md:grid-cols-2 lg:grid-cols-3">
|
||
{searchedProjects.map((projectId) => {
|
||
const projectDetails = getProjectById(projectId);
|
||
|
||
if (!projectDetails) return;
|
||
|
||
return <ProjectCard key={projectDetails.id} project={projectDetails} />;
|
||
})}
|
||
</div>
|
||
)}
|
||
</div>
|
||
) : (
|
||
<EmptyState
|
||
image={emptyStateImage}
|
||
title="Start a Project"
|
||
description="Think of each project as the parent for goal-oriented work. Projects are where Jobs, Cycles, and Modules live and, along with your colleagues, help you achieve that goal."
|
||
primaryButton={{
|
||
text: "Start your first project",
|
||
onClick: () => {
|
||
setTrackElement("Project empty state");
|
||
commandPaletteStore.toggleCreateProjectModal(true);
|
||
},
|
||
}}
|
||
comicBox={{
|
||
title: "Everything starts with a project in Plane",
|
||
description: "A project could be a product’s roadmap, a marketing campaign, or launching a new car.",
|
||
}}
|
||
size="lg"
|
||
disabled={!isEditingAllowed}
|
||
/>
|
||
)}
|
||
</>
|
||
);
|
||
});
|