added project page to org dialog

moved sprint creation/management there
This commit is contained in:
Oliver Bryan
2026-01-12 23:17:25 +00:00
parent 5550030743
commit 8eb65173a6
5 changed files with 202 additions and 77 deletions

View File

@@ -116,13 +116,19 @@ export function CreateIssue({
console.error(actionErr);
}
},
onError: (error) => {
onError: async (error) => {
setError(error);
setSubmitting(false);
toast.error(`Error creating issue: ${error}`, {
dismissible: false,
});
try {
await errorAction?.(error);
} catch (actionErr) {
console.error(actionErr);
}
},
});
} catch (err) {

View File

@@ -3,14 +3,19 @@ import {
ISSUE_STATUS_MAX_LENGTH,
type OrganisationMemberResponse,
type OrganisationResponse,
type ProjectRecord,
type ProjectResponse,
type SprintRecord,
} from "@issue/shared";
import { ChevronDown, ChevronUp, EllipsisVertical, Plus, X } from "lucide-react";
import type { ReactNode } from "react";
import { useCallback, useEffect, useState } from "react";
import { type ReactNode, useCallback, useEffect, useState } from "react";
import { toast } from "sonner";
import { AddMemberDialog } from "@/components/add-member-dialog";
import { CreateSprint } from "@/components/create-sprint";
import { OrganisationSelect } from "@/components/organisation-select";
import { ProjectSelect } from "@/components/project-select";
import { useAuthenticatedSession } from "@/components/session-provider";
import SmallSprintDisplay from "@/components/small-sprint-display";
import SmallUserDisplay from "@/components/small-user-display";
import StatusTag from "@/components/status-tag";
import { Button } from "@/components/ui/button";
@@ -35,12 +40,24 @@ function OrganisationsDialog({
selectedOrganisation,
setSelectedOrganisation,
refetchOrganisations,
projects,
selectedProject,
sprints,
onSelectedProjectChange,
onCreateProject,
onCreateSprint,
}: {
trigger?: ReactNode;
organisations: OrganisationResponse[];
selectedOrganisation: OrganisationResponse | null;
setSelectedOrganisation: (organisation: OrganisationResponse | null) => void;
refetchOrganisations: (options?: { selectOrganisationId?: number }) => Promise<void>;
projects: ProjectResponse[];
selectedProject: ProjectResponse | null;
sprints: SprintRecord[];
onSelectedProjectChange: (project: ProjectResponse | null) => void;
onCreateProject: (project: ProjectRecord) => void | Promise<void>;
onCreateSprint: (sprint: SprintRecord) => void | Promise<void>;
}) {
const { user } = useAuthenticatedSession();
@@ -79,6 +96,20 @@ function OrganisationsDialog({
selectedOrganisation?.OrganisationMember.role === "owner" ||
selectedOrganisation?.OrganisationMember.role === "admin";
const formatDate = (value: Date | string) =>
new Date(value).toLocaleDateString(undefined, { month: "short", day: "numeric" });
const getSprintDateRange = (sprint: SprintRecord) => {
if (!sprint.startDate || !sprint.endDate) return "";
return `${formatDate(sprint.startDate)} - ${formatDate(sprint.endDate)}`;
};
const isCurrentSprint = (sprint: SprintRecord) => {
if (!sprint.startDate || !sprint.endDate) return false;
const today = new Date();
const start = new Date(sprint.startDate);
const end = new Date(sprint.endDate);
return start <= today && today <= end;
};
const refetchMembers = useCallback(async () => {
if (!selectedOrganisation) return;
try {
@@ -436,15 +467,15 @@ function OrganisationsDialog({
)}
</DialogTrigger>
<DialogContent className="max-w-lg w-full max-w-[calc(100vw-2rem)]">
<DialogContent className="max-w-sm">
<DialogHeader>
<DialogTitle>Organisations</DialogTitle>
</DialogHeader>
<div className="flex flex-col gap-2">
{selectedOrganisation ? (
<Tabs value={activeTab} onValueChange={setActiveTab} className="w-full">
<div className="flex gap-2 items-center">
<Tabs value={activeTab} onValueChange={setActiveTab} className="w-full min-w-0">
<div className="flex flex-wrap gap-2 items-center w-full min-w-0">
<OrganisationSelect
organisations={organisations}
selectedOrganisation={selectedOrganisation}
@@ -468,6 +499,7 @@ function OrganisationsDialog({
<TabsList>
<TabsTrigger value="info">Info</TabsTrigger>
<TabsTrigger value="users">Users</TabsTrigger>
<TabsTrigger value="projects">Projects</TabsTrigger>
<TabsTrigger value="issues">Issues</TabsTrigger>
</TabsList>
</div>
@@ -582,6 +614,88 @@ function OrganisationsDialog({
</div>
</TabsContent>
<TabsContent value="projects">
<div className="border p-2 min-w-0 overflow-hidden">
<div className="flex flex-col gap-3">
<ProjectSelect
projects={projects}
selectedProject={selectedProject}
organisationId={selectedOrganisation?.Organisation.id}
onSelectedProjectChange={onSelectedProjectChange}
onCreateProject={onCreateProject}
showLabel
/>
<div className="flex gap-3 flex-col">
<div className="border p-2 min-w-0 overflow-hidden">
{selectedProject ? (
<>
<h2 className="text-xl font-600 mb-2 break-all">
{selectedProject.Project.name}
</h2>
<div className="flex flex-col gap-1">
<p className="text-sm text-muted-foreground break-all">
Key: {selectedProject.Project.key}
</p>
<p className="text-sm text-muted-foreground break-all">
Creator: {selectedProject.User.name}
</p>
</div>
</>
) : (
<p className="text-sm text-muted-foreground">
Select a project to view details.
</p>
)}
</div>
<div className="flex flex-col gap-2 min-w-0 flex-1">
{selectedProject ? (
<div className="flex flex-col gap-2 max-h-56 overflow-y-scroll">
{sprints.map((sprint) => {
const dateRange = getSprintDateRange(sprint);
const isCurrent = isCurrentSprint(sprint);
return (
<div
key={sprint.id}
className={`flex items-center justify-between p-2 border ${
isCurrent
? "border-emerald-500/60 bg-emerald-500/10"
: ""
}`}
>
<SmallSprintDisplay sprint={sprint} />
{dateRange && (
<span className="text-xs text-muted-foreground">
{dateRange}
</span>
)}
</div>
);
})}
{isAdmin && (
<CreateSprint
projectId={selectedProject?.Project.id}
completeAction={onCreateSprint}
trigger={
<Button variant="outline" size="sm">
Create sprint{" "}
<Plus className="size-4" />
</Button>
}
/>
)}
</div>
) : (
<p className="text-sm text-muted-foreground">
Select a project to view sprints.
</p>
)}
</div>
</div>
</div>
</div>
</TabsContent>
<TabsContent value="issues">
<div className="border p-2 min-w-0 overflow-hidden">
<h2 className="text-xl font-600 mb-2">Issue Statuses</h2>
@@ -659,7 +773,7 @@ function OrganisationsDialog({
{isAdmin &&
(isCreatingStatus ? (
<>
<div className="flex gap-2">
<div className="flex gap-2 w-full min-w-0">
<Input
value={newStatusName}
maxLength={ISSUE_STATUS_MAX_LENGTH}
@@ -668,7 +782,7 @@ function OrganisationsDialog({
if (statusError) setStatusError(null);
}}
placeholder="Status name"
className="flex-1"
className="flex-1 w-0 min-w-0"
onKeyDown={(e) => {
if (e.key === "Enter") {
void handleCreateStatus();
@@ -712,6 +826,7 @@ function OrganisationsDialog({
setIsCreatingStatus(true);
setStatusError(null);
}}
className="flex gap-2 w-full min-w-0"
>
Create status <Plus className="size-4" />
</Button>
@@ -721,7 +836,7 @@ function OrganisationsDialog({
</TabsContent>
</Tabs>
) : (
<div className="flex flex-col gap-2">
<div className="flex flex-col gap-2 w-full min-w-0">
<OrganisationSelect
organisations={organisations}
selectedOrganisation={selectedOrganisation}

View File

@@ -49,7 +49,7 @@ function DialogContent({
"bg-background data-[state=closed]:zoom-out-95",
"data-[state=open]:zoom-in-95 fixed top-[50%] left-[50%]",
"z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%]",
"gap-4 border p-4 shadow-lg duration-200 outline-none sm:max-w-lg",
"gap-4 border p-4 shadow-lg duration-200 outline-none",
className,
)}
{...props}

View File

@@ -4,6 +4,7 @@ import type {
IssueResponse,
OrganisationMemberResponse,
OrganisationResponse,
ProjectRecord,
ProjectResponse,
SprintRecord,
UserRecord,
@@ -12,7 +13,6 @@ import { useEffect, useMemo, useRef, useState } from "react";
import { toast } from "sonner";
import AccountDialog from "@/components/account-dialog";
import { CreateIssue } from "@/components/create-issue";
import { CreateSprint } from "@/components/create-sprint";
import { IssueDetailPane } from "@/components/issue-detail-pane";
import { IssuesTable } from "@/components/issues-table";
import LogOutButton from "@/components/log-out-button";
@@ -53,10 +53,6 @@ export default function App() {
const [members, setMembers] = useState<UserRecord[]>([]);
const [sprints, setSprints] = useState<SprintRecord[]>([]);
const isAdmin =
selectedOrganisation?.OrganisationMember.role === "owner" ||
selectedOrganisation?.OrganisationMember.role === "admin";
const deepLinkParams = useMemo(() => {
const params = new URLSearchParams(window.location.search);
const orgSlug = params.get("o")?.trim().toLowerCase() ?? "";
@@ -394,6 +390,43 @@ export default function App() {
}
}, [deepLinkParams, selectedOrganisation, selectedProject]);
const handleProjectChange = (project: ProjectResponse | null) => {
setSelectedProject(project);
localStorage.setItem("selectedProjectId", `${project?.Project.id}`);
setSelectedIssue(null);
updateUrlParams({
projectKey: project?.Project.key.toLowerCase() ?? null,
issueNumber: null,
});
};
const handleProjectCreate = async (project: ProjectRecord) => {
if (!selectedOrganisation) return;
toast.success(`Created Project ${project.name}`, {
dismissible: false,
});
await refetchProjects(selectedOrganisation.Organisation.id, {
selectProjectId: project.id,
});
};
const handleSprintCreate = async (sprint: SprintRecord) => {
if (!selectedProject) return;
toast.success(
<>
Created sprint <span style={{ color: sprint.color }}>{sprint.name}</span>
</>,
{
dismissible: false,
},
);
await refetchSprints(selectedProject.Project.id);
};
return (
<main className={`w-full h-screen flex flex-col gap-${BREATHING_ROOM} p-${BREATHING_ROOM}`}>
{/* header area */}
@@ -427,71 +460,33 @@ export default function App() {
projects={projects}
selectedProject={selectedProject}
organisationId={selectedOrganisation?.Organisation.id}
onSelectedProjectChange={(project) => {
setSelectedProject(project);
localStorage.setItem("selectedProjectId", `${project?.Project.id}`);
setSelectedIssue(null);
updateUrlParams({
projectKey: project?.Project.key.toLowerCase() ?? null,
issueNumber: null,
});
}}
onCreateProject={async (project) => {
if (!selectedOrganisation) return;
toast.success(`Created Project ${project.name}`, {
dismissible: false,
});
await refetchProjects(selectedOrganisation.Organisation.id, {
selectProjectId: project.id,
});
}}
onSelectedProjectChange={handleProjectChange}
onCreateProject={handleProjectCreate}
showLabel
/>
)}
{selectedOrganisation && selectedProject && (
<>
<CreateIssue
projectId={selectedProject?.Project.id}
sprints={sprints}
members={members}
statuses={selectedOrganisation.Organisation.statuses}
completeAction={async (issueNumber) => {
if (!selectedProject) return;
toast.success(
`Created ${issueID(selectedProject.Project.key, issueNumber)}`,
{
dismissible: false,
},
);
await refetchIssues();
}}
errorAction={async (errorMessage) => {
toast.error(`Error creating issue: ${errorMessage}`, {
<CreateIssue
projectId={selectedProject?.Project.id}
sprints={sprints}
members={members}
statuses={selectedOrganisation.Organisation.statuses}
completeAction={async (issueNumber) => {
if (!selectedProject) return;
toast.success(
`Created ${issueID(selectedProject.Project.key, issueNumber)}`,
{
dismissible: false,
});
}}
/>
{isAdmin && (
<CreateSprint
projectId={selectedProject?.Project.id}
completeAction={async (sprint) => {
if (!selectedProject) return;
toast.success(
<>
Created sprint{" "}
<span style={{ color: sprint.color }}>{sprint.name}</span>
</>,
{
dismissible: false,
},
);
await refetchSprints(selectedProject?.Project.id);
}}
/>
)}
</>
},
);
await refetchIssues();
}}
errorAction={async (errorMessage) => {
toast.error(`Error creating issue: ${errorMessage}`, {
dismissible: false,
});
}}
/>
)}
</div>
<div className={`flex gap-${BREATHING_ROOM} items-center`}>
@@ -510,6 +505,12 @@ export default function App() {
selectedOrganisation={selectedOrganisation}
setSelectedOrganisation={setSelectedOrganisation}
refetchOrganisations={refetchOrganisations}
projects={projects}
selectedProject={selectedProject}
sprints={sprints}
onSelectedProjectChange={handleProjectChange}
onCreateProject={handleProjectCreate}
onCreateSprint={handleSprintCreate}
/>
</DropdownMenuItem>
<DropdownMenuItem asChild className="flex items-end justify-end">