# HANDOFF — Brief 5 phút cho session tiếp theo **Last updated:** 2026-05-07 (Session 15 — **Tooltip diagnose "Lưu & Gửi Duyệt" silent disabled (commit `835cc7f`). Plan drastic refactor flat workflow user chốt → attempt 12 file edits Domain/Configurations, realize scope ~8-10h vượt session, REVERT clean. State 96 test pass, 20 mig.**) ## TL;DR Session 15 (07/05 — Tooltip diagnose + drastic refactor DEFER) User UAT live screenshot báo button "Lưu & Gửi Duyệt" KHÔNG hoạt động + suy đoán "trùng ID" giữa các phiếu. **Diagnose (commit `835cc7f`):** - Root cause: button silent disabled khi `evaluation.workflow.nextPhases` không có forward phase (chỉ TuChoi/TraLai). Cause khả năng: workflow definition pinned thiếu adjacent step → `policy.NextPhasesFrom(DangSoanThao)` return empty. - Improvement: tooltip + dialog hiển thị reason rõ ràng: - `submitDisabledReason` text: "Phiếu đã ở phase X — chỉ Bản nháp/Trả lại mới sửa+gửi" / "Workflow không có phase tiếp theo từ X. Liên hệ admin kiểm tra cấu hình" - Button title attribute → hover show reason hoặc forward phase label - Dialog confirm show forward phase explicit ("Sẽ chuyển sang Chờ Purchasing") - Mirror fe-admin + fe-user. Build pass cả 2. KHÔNG đụng BE — chỉ FE diagnostic UX. - "Trùng ID" KHÔNG phải bug FE — `PurchaseEvaluationWorkspacePage` URL state đúng, mỗi PE row unique GUID + MaPhieu. Suy đoán user do button silent. **Plan drastic refactor → DEFER:** User confirm "bỏ phase enum hoàn toàn, dùng ChoDuyet=10 đơn nhất + currentStepIndex tracking" — refactor workflow từ phase-based + InnerStep nested model sang flat WorkflowStep model (mỗi step = Phòng × Cấp + Approvers users). Edit working tree 12 files (Domain entities + EF Configurations + DbContext): - Phase enum +ChoDuyet=10, legacy values 2-6 deprecated - WorkflowStep +DepartmentId, +PositionLevel - Drop class WorkflowStepInnerStep + nav (PE + Contract) - PE/Contract +CurrentWorkflowStepIndex int?, +RejectedAtStepIndex int? - *DepartmentApproval drop InnerStepId column - EF Configurations: drop InnerStep config + nav, restore simple unique non-filtered - DbContext: drop DbSet × 2 Reality check scope realistic ~8-10h: 1. Domain + EF + DbContext (~50min) ✓ done in working tree 2. PolicyRegistry rewrite PE+Contract (~45min) 3. App CQRS DTOs rewrite (~45min) 4. Service rewrite PE+Contract (~2-3h) 5. Tests rewrite — drop 12 N-stage tests + update remaining (~1.5h) 6. Migration 21 + LocalDB apply + verify (~30min) 7. FE Designer rewrite (~1.5h) 8. FE PeWorkflowPanel + workflow timeline (~1h) 9. Docs/Skill update (~45min) Vượt session boundary + risk session deep ~30 commits → **REVERT working tree** về `835cc7f` clean state. Test 96 pass intact. **Decision memorized:** add memory `feedback_drastic_refactor_scope.md` — drastic refactor cần dedicated session, scope estimation conservative (2x buffer), tránh mid-session big refactor. ## ⚠️ CẢNH BÁO Session 16+ 1. **Drastic refactor flat workflow chưa làm — DEFER** với plan chi tiết. Khi resume: - Plan kỹ 6 chunk per-commit - Buffer 2x estimate (~16h thực tế) - Tests rewrite biggest risk - Hoặc fall back Approach Y (FE Designer flat UI giới hạn 5 phòng) ROI 1-2h nếu user OK trade-off 2. **Task 2 sample data seed N-stage** vẫn pending (block trên DesignTime vs Runtime DB gotcha + DbInitializer seed flow) 3. **schema-diagram §17-19 Mig 18-20** vẫn defer cron audit 2026-06-01 4. **Hard blockers Ops** giữ nguyên 6 task ## TL;DR Session 14 (07/05 — PE 3-button approval workflow) User chỉ thị thay 2-button approval (Duyệt + Reject mơ hồ) bằng **3 hành động rõ ràng** cho approver: - **Duyệt** = forward phase tiếp theo (decision=Approve) - **Trả lại** = về DangSoanThao + Drafter sửa (decision=Reject + target=DangSoanThao). Smart reject pattern Mig 16 + clear N-stage rows + Drafter resume jump-back tới phase đã reject. - **Từ chối** = phase=TuChoi (decision=Reject + target=TuChoi). Phiếu khoá vĩnh viễn (17 handler Mig 16 lock edit). Drafter phải tạo phiếu mới. **1 commit (`0d77698`):** - **Domain `PurchaseEvaluationPolicy.cs`**: NccOnly + NccWithPlan thêm `(X → TuChoi)` transition cho mọi phase trung gian (ChoPurchasing/ChoDuAn/ChoCCM/ChoCEODuyetPA/ChoCEODuyetNCC) với roles của phase đó. FromDefinition expand: mỗi step (trừ DangSoanThao) thêm (step.Phase → TuChoi) với roles step. - **Service** `PurchaseEvaluationWorkflowService.TransitionAsync` — Reject branch tách 2 case: ``` if (decision == Reject) { if (target != TuChoi) { // Trả lại RejectedFromPhase = fromPhase target = DangSoanThao // force clear N-stage rows tại fromPhase } // else target=TuChoi: giữ nguyên, KHÔNG set RejectedFromPhase, KHÔNG clear } ``` - **FE PeWorkflowPanel (admin + user mirror)**: render 3 button rõ: - "✓ Duyệt →