修复 WUI Kanban 看板选择与隔离 (#594)

* fix: add explicit kanban board selection

* fix: tighten kanban board counts and management
This commit is contained in:
Zhicheng Han
2026-05-10 13:58:44 +02:00
committed by GitHub
parent 377fa4144d
commit 838791a740
17 changed files with 1199 additions and 156 deletions
+220 -30
View File
@@ -1,83 +1,259 @@
import { defineStore } from 'pinia'
import { ref } from 'vue'
import { computed, ref } from 'vue'
import * as kanbanApi from '@/api/hermes/kanban'
import type { KanbanTask, KanbanStats, KanbanAssignee } from '@/api/hermes/kanban'
import type { KanbanTask, KanbanStats, KanbanAssignee, KanbanBoard, KanbanCapabilities } from '@/api/hermes/kanban'
export const KANBAN_SELECTED_BOARD_STORAGE_KEY = 'hermes.kanban.selectedBoard'
export const DEFAULT_KANBAN_BOARD = 'default'
const BOARD_SLUG_RE = /^[a-z0-9][a-z0-9-]{0,62}$/
function safeStorageGet(key: string): string | null {
if (typeof window === 'undefined') return null
try {
return window.localStorage.getItem(key)
} catch {
return null
}
}
function safeStorageSet(key: string, value: string) {
if (typeof window === 'undefined') return
try {
window.localStorage.setItem(key, value)
} catch {
// Ignore storage failures; selectedBoard still remains explicit in-memory.
}
}
export function normalizeBoardSlug(board?: string | null): string {
const trimmed = board?.trim()
if (!trimmed) return DEFAULT_KANBAN_BOARD
return BOARD_SLUG_RE.test(trimmed) ? trimmed : DEFAULT_KANBAN_BOARD
}
export const useKanbanStore = defineStore('kanban', () => {
const tasks = ref<KanbanTask[]>([])
const stats = ref<KanbanStats | null>(null)
const assignees = ref<KanbanAssignee[]>([])
const boards = ref<KanbanBoard[]>([])
const capabilities = ref<KanbanCapabilities | null>(null)
const loading = ref(false)
const boardsLoading = ref(false)
const boardWarning = ref<string | null>(null)
const selectedBoard = ref(normalizeBoardSlug(safeStorageGet(KANBAN_SELECTED_BOARD_STORAGE_KEY)))
const filterStatus = ref<string | null>(null)
const filterAssignee = ref<string | null>(null)
let boardGeneration = 0
let boardsRequestSeq = 0
let tasksRequestSeq = 0
let statsRequestSeq = 0
let assigneesRequestSeq = 0
let loadingRequestSeq = 0
const activeBoards = computed(() => {
const visible = boards.value.filter(board => !board.archived)
if (!visible.some(board => board.slug === DEFAULT_KANBAN_BOARD)) {
return [{
slug: DEFAULT_KANBAN_BOARD,
name: 'Default',
description: '',
icon: '',
color: '',
created_at: null,
archived: false,
counts: {},
total: 0,
}, ...visible]
}
return visible
})
function boardExists(board: string): boolean {
return activeBoards.value.some(item => item.slug === board)
}
function resolveAvailableBoard(candidate?: string | null): string {
const normalized = normalizeBoardSlug(candidate)
if (boards.value.length > 0 && !boardExists(normalized)) return DEFAULT_KANBAN_BOARD
return normalized
}
function clearBoardScopedState() {
tasks.value = []
stats.value = null
assignees.value = []
}
function setSelectedBoard(board?: string | null): string {
const resolved = resolveAvailableBoard(board)
const changed = selectedBoard.value !== resolved
selectedBoard.value = resolved
safeStorageSet(KANBAN_SELECTED_BOARD_STORAGE_KEY, resolved)
boardWarning.value = null
if (changed) {
clearBoardScopedState()
boardGeneration++
}
return resolved
}
function recoverSelectedBoard(candidate?: string | null): { board: string; recovered: boolean } {
const normalized = normalizeBoardSlug(candidate)
const resolved = resolveAvailableBoard(normalized)
const recovered = resolved !== normalized
setSelectedBoard(resolved)
if (recovered) {
boardWarning.value = `Board "${normalized}" is unavailable; fell back to "${resolved}".`
}
return { board: resolved, recovered }
}
function nextRequestContext(nextSeq: () => number) {
const seq = nextSeq()
const generation = boardGeneration
const board = selectedBoard.value
return { seq, generation, board }
}
function isCurrentRequest(seq: number, generation: number, board: string, currentSeq: number): boolean {
return seq === currentSeq && generation === boardGeneration && board === selectedBoard.value
}
async function fetchBoards(includeArchived = false) {
const seq = ++boardsRequestSeq
boardsLoading.value = true
try {
const nextBoards = await kanbanApi.listBoards({ includeArchived })
if (seq !== boardsRequestSeq) return
boards.value = nextBoards
const resolved = resolveAvailableBoard(selectedBoard.value)
if (resolved !== selectedBoard.value) recoverSelectedBoard(selectedBoard.value)
} catch (err) {
if (seq === boardsRequestSeq) console.error('Failed to fetch kanban boards:', err)
} finally {
if (seq === boardsRequestSeq) boardsLoading.value = false
}
}
async function fetchCapabilities() {
try {
capabilities.value = await kanbanApi.getCapabilities()
} catch (err) {
console.error('Failed to fetch kanban capabilities:', err)
}
}
async function createBoard(data: { slug: string; name?: string; description?: string; icon?: string; color?: string; switchCurrent?: boolean }) {
const board = await kanbanApi.createBoard(data)
await fetchBoards()
setSelectedBoard(board.slug)
await refreshAll()
return board
}
async function archiveSelectedBoard() {
const board = selectedBoard.value
if (board === DEFAULT_KANBAN_BOARD) throw new Error('Cannot archive the default kanban board')
await kanbanApi.archiveBoard(board)
await fetchBoards()
setSelectedBoard(DEFAULT_KANBAN_BOARD)
await refreshAll()
}
async function fetchTasks(silent = false) {
const { seq, generation, board } = nextRequestContext(() => ++tasksRequestSeq)
const loadingSeq = silent ? 0 : ++loadingRequestSeq
if (!silent) loading.value = true
try {
tasks.value = await kanbanApi.listTasks({
const nextTasks = await kanbanApi.listTasks({
board,
status: filterStatus.value || undefined,
assignee: filterAssignee.value || undefined,
})
if (isCurrentRequest(seq, generation, board, tasksRequestSeq)) tasks.value = nextTasks
} catch (err) {
console.error('Failed to fetch kanban tasks:', err)
if (isCurrentRequest(seq, generation, board, tasksRequestSeq)) console.error('Failed to fetch kanban tasks:', err)
} finally {
if (!silent) loading.value = false
if (!silent && loadingSeq === loadingRequestSeq) loading.value = false
}
}
async function fetchStats() {
const { seq, generation, board } = nextRequestContext(() => ++statsRequestSeq)
try {
stats.value = await kanbanApi.getStats()
const nextStats = await kanbanApi.getStats({ board })
if (isCurrentRequest(seq, generation, board, statsRequestSeq)) stats.value = nextStats
} catch (err) {
console.error('Failed to fetch kanban stats:', err)
if (isCurrentRequest(seq, generation, board, statsRequestSeq)) console.error('Failed to fetch kanban stats:', err)
}
}
async function fetchAssignees() {
const { seq, generation, board } = nextRequestContext(() => ++assigneesRequestSeq)
try {
assignees.value = await kanbanApi.getAssignees()
const nextAssignees = await kanbanApi.getAssignees({ board })
if (isCurrentRequest(seq, generation, board, assigneesRequestSeq)) assignees.value = nextAssignees
} catch (err) {
console.error('Failed to fetch kanban assignees:', err)
if (isCurrentRequest(seq, generation, board, assigneesRequestSeq)) console.error('Failed to fetch kanban assignees:', err)
}
}
async function createTask(data: { title: string; body?: string; assignee?: string; priority?: number; tenant?: string }) {
const task = await kanbanApi.createTask(data)
tasks.value.unshift(task)
await fetchStats()
const board = selectedBoard.value
const task = await kanbanApi.createTask(data, { board })
if (board === selectedBoard.value) {
tasks.value.unshift(task)
await Promise.all([fetchStats(), fetchBoards()])
}
return task
}
async function completeTasks(taskIds: string[], summary?: string) {
await kanbanApi.completeTasks(taskIds, summary)
for (const id of taskIds) {
const task = tasks.value.find(t => t.id === id)
if (task) task.status = 'done'
const board = selectedBoard.value
await kanbanApi.completeTasks(taskIds, summary, { board })
if (board === selectedBoard.value) {
for (const id of taskIds) {
const task = tasks.value.find(t => t.id === id)
if (task) task.status = 'done'
}
await Promise.all([fetchStats(), fetchBoards()])
}
await fetchStats()
}
async function blockTask(taskId: string, reason: string) {
await kanbanApi.blockTask(taskId, reason)
const task = tasks.value.find(t => t.id === taskId)
if (task) task.status = 'blocked'
await fetchStats()
const board = selectedBoard.value
await kanbanApi.blockTask(taskId, reason, { board })
if (board === selectedBoard.value) {
const task = tasks.value.find(t => t.id === taskId)
if (task) task.status = 'blocked'
await Promise.all([fetchStats(), fetchBoards()])
}
}
async function unblockTasks(taskIds: string[]) {
await kanbanApi.unblockTasks(taskIds)
for (const id of taskIds) {
const task = tasks.value.find(t => t.id === id)
if (task) task.status = 'ready'
const board = selectedBoard.value
await kanbanApi.unblockTasks(taskIds, { board })
if (board === selectedBoard.value) {
for (const id of taskIds) {
const task = tasks.value.find(t => t.id === id)
if (task) task.status = 'ready'
}
await Promise.all([fetchStats(), fetchBoards()])
}
await fetchStats()
}
async function assignTask(taskId: string, profile: string) {
await kanbanApi.assignTask(taskId, profile)
const task = tasks.value.find(t => t.id === taskId)
if (task) task.assignee = profile
const board = selectedBoard.value
await kanbanApi.assignTask(taskId, profile, { board })
if (board === selectedBoard.value) {
const task = tasks.value.find(t => t.id === taskId)
if (task) task.assignee = profile
await Promise.all([fetchStats(), fetchAssignees()])
}
}
function setFilter(key: 'status' | 'assignee', value: string | null) {
@@ -86,25 +262,39 @@ export const useKanbanStore = defineStore('kanban', () => {
}
async function refreshAll() {
await Promise.all([fetchTasks(), fetchStats(), fetchAssignees()])
await Promise.all([fetchBoards(), fetchTasks(), fetchStats(), fetchAssignees()])
}
return {
tasks,
stats,
assignees,
boards,
capabilities,
activeBoards,
loading,
boardsLoading,
boardWarning,
selectedBoard,
filterStatus,
filterAssignee,
fetchBoards,
fetchCapabilities,
fetchTasks,
fetchStats,
fetchAssignees,
createTask,
createBoard,
archiveSelectedBoard,
completeTasks,
blockTask,
unblockTasks,
assignTask,
setFilter,
setSelectedBoard,
recoverSelectedBoard,
resolveAvailableBoard,
clearBoardScopedState,
refreshAll,
}
})