144a8a4685
LST-69 (3.2) front. Client portal UI on the phase-1 backend. - New frontend/modules/client-portal/ layer: /portal (project cards from the client's allowedProjects via /me), /portal/projects/[id] (tickets list, detail modal, create modal with document upload), client-tickets service + DTO, CT-XXX formatting. - Front tenancy: auth.global.ts redirects a pure ROLE_CLIENT to /portal and blocks internal routes; portal pages open to any authenticated user. - Admin: UserDrawer manages client accounts (ROLE_CLIENT + client + allowedProjects); new "Tickets client" admin tab (list, filters, status change with required comment on reject, detail modal). - Kanban/my-tasks: client-ticket icon + tooltip when task.clientTicket is set (data via task:read, no extra call). TaskDocument upload generalized with a clientTicketId prop. getContent uses native fetch (text response). - i18n portal/clientTicket keys; sidebar /portal item (module client-portal). nuxt build passes; /portal routes present, existing routes intact.
35 lines
937 B
TypeScript
35 lines
937 B
TypeScript
import type { TaskDocument } from '~/modules/project-management/services/dto/task-document'
|
|
|
|
export type ClientTicketType = 'bug' | 'improvement' | 'other'
|
|
export type ClientTicketStatus = 'new' | 'in_progress' | 'done' | 'rejected'
|
|
|
|
export type ClientTicket = {
|
|
'@id'?: string
|
|
id: number
|
|
number: number
|
|
type: ClientTicketType
|
|
title: string
|
|
description: string
|
|
url: string | null
|
|
status: ClientTicketStatus
|
|
statusComment: string | null
|
|
project: string // IRI
|
|
submittedBy: string | null // IRI, nullable (ON DELETE SET NULL)
|
|
createdAt: string
|
|
updatedAt: string
|
|
documents?: TaskDocument[]
|
|
}
|
|
|
|
export type ClientTicketCreate = {
|
|
type: ClientTicketType
|
|
title: string
|
|
description: string
|
|
url?: string | null
|
|
project: string // IRI
|
|
}
|
|
|
|
export type ClientTicketStatusUpdate = {
|
|
status: ClientTicketStatus
|
|
statusComment?: string | null
|
|
}
|