* chore(ST-1112): add deps for task dialog Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): add shared zod schema for task dialog Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): add missing MD3 tokens for task dialog outline-variant, on-error-container, status-review (light + dark) Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): add saveTask and deleteTask server actions for TaskDialog Unified create/edit action (saveTask) replaces separate formData-based actions for the new TaskDialog. Uses shared zod schema, structured SaveTaskResult union type, and context-aware revalidatePath for both sprint and backlog routes. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): add TaskDialog component (create & edit mode) Builds the full TaskDialog on top of the existing @base-ui/react Dialog primitive. Covers create mode, edit mode (status field + created_at metadata + delete), dirty-check AlertDialog, delete confirm AlertDialog, Cmd+Enter submit, and per-field char counters. Uses react-hook-form + zodResolver against the shared taskSchema. Priority and status are extracted to PrioritySegmented and StatusSelect sub-components using MD3 tokens throughout. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): refactor task-list to open TaskDialog via URL params Replaces inline create/edit forms with router.push navigation: - Clicking a task row → ?editTask=<id> - "+ Taak" button → ?newTask=1&storyId=<storyId> Removes CreateTaskForm, EditSubmitButton, updateTaskAction, and createTaskAction from the component. Status toggle and DnD remain unchanged. Rows now have cursor-pointer and keyboard a11y. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): wire TaskDialog into sprint page via searchParams Sprint page now reads ?newTask, ?storyId, and ?editTask query params. For edit mode: fetches the task server-side with productAccessFilter scope (invalid/foreign IDs redirect to closePath). Renders TaskDialog when either param is present. closePath is the sprint route without query params. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): add Suspense skeleton for edit-mode task loading Extracts task fetch into EditTaskLoader (async server component) so the sprint board renders immediately while the task loads. TaskDialogSkeleton shows 3 grey bars during the fetch. Invalid or out-of-scope task IDs redirect to closePath. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): render description as markdown in task-detail-dialog Solo task detail now renders description via react-markdown + remark-gfm with prose styling. Sanitizes script/iframe elements. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * test(ST-1112): add saveTask/deleteTask server action tests Covers all three demo-policy layers and cross-tenant scope: demo blocked (403), unauthenticated blocked, validation 422, edit cross-tenant forbidden, create cross-tenant forbidden, and happy-path for both edit and create. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): add updateTaskStatusWithStoryPromotion helper Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1112): wire story-promotion into saveTask and PATCH /api/tasks/:id Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * docs(ST-1112): add task-dialog doc and architecture note Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * chore: extend allowed tools in settings.local.json Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1113): add 200ms animation-delay to TaskDialogSkeleton to prevent flicker Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1114): add DirtyCloseGuard reusable component for dirty-form close confirmation Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * feat(ST-1114): add shared Markdown wrapper, apply to task-detail and story-dialog Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> * chore: allow grep -E pattern in settings.local.json Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> --------- Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
72 lines
2.2 KiB
TypeScript
72 lines
2.2 KiB
TypeScript
import type { Prisma, TaskStatus } from '@prisma/client'
|
|
import { prisma } from '@/lib/prisma'
|
|
|
|
export type StoryStatusChange = 'promoted' | 'demoted' | null
|
|
|
|
export interface UpdateTaskStatusResult {
|
|
task: {
|
|
id: string
|
|
title: string
|
|
status: TaskStatus
|
|
story_id: string
|
|
implementation_plan: string | null
|
|
}
|
|
storyStatusChange: StoryStatusChange
|
|
storyId: string
|
|
}
|
|
|
|
// Update task.status atomically and auto-promote/demote the parent story:
|
|
// - All sibling tasks DONE → story.status = DONE
|
|
// - Story was DONE and a task moves out of DONE → story.status = IN_SPRINT
|
|
// Demote target is IN_SPRINT (not OPEN): OPEN means "back in product backlog",
|
|
// which is a sprint-management action, not a status side-effect.
|
|
export async function updateTaskStatusWithStoryPromotion(
|
|
taskId: string,
|
|
newStatus: TaskStatus,
|
|
client?: Prisma.TransactionClient,
|
|
): Promise<UpdateTaskStatusResult> {
|
|
const run = async (tx: Prisma.TransactionClient): Promise<UpdateTaskStatusResult> => {
|
|
const task = await tx.task.update({
|
|
where: { id: taskId },
|
|
data: { status: newStatus },
|
|
select: {
|
|
id: true,
|
|
title: true,
|
|
status: true,
|
|
story_id: true,
|
|
implementation_plan: true,
|
|
},
|
|
})
|
|
|
|
const siblings = await tx.task.findMany({
|
|
where: { story_id: task.story_id },
|
|
select: { status: true },
|
|
})
|
|
const allDone = siblings.every((s) => s.status === 'DONE')
|
|
|
|
const story = await tx.story.findUniqueOrThrow({
|
|
where: { id: task.story_id },
|
|
select: { status: true },
|
|
})
|
|
|
|
let storyStatusChange: StoryStatusChange = null
|
|
if (newStatus === 'DONE' && allDone && story.status !== 'DONE') {
|
|
await tx.story.update({
|
|
where: { id: task.story_id },
|
|
data: { status: 'DONE' },
|
|
})
|
|
storyStatusChange = 'promoted'
|
|
} else if (newStatus !== 'DONE' && story.status === 'DONE') {
|
|
await tx.story.update({
|
|
where: { id: task.story_id },
|
|
data: { status: 'IN_SPRINT' },
|
|
})
|
|
storyStatusChange = 'demoted'
|
|
}
|
|
|
|
return { task, storyStatusChange, storyId: task.story_id }
|
|
}
|
|
|
|
if (client) return run(client)
|
|
return prisma.$transaction(run)
|
|
}
|