test(frontend): raise overall statement coverage to 30.42% for Phase 1 MVP

This commit is contained in:
2026-06-13 22:33:11 +07:00
parent 190b9a3af5
commit 9c5df0abdb
37 changed files with 6128 additions and 24 deletions
@@ -0,0 +1,119 @@
// File: frontend/components/correspondences/circulation-status-card.test.tsx
// Change Log:
// - 2026-06-13: Initial creation - test coverage for circulation-status-card component
import { describe, it, expect, vi, beforeEach } from 'vitest';
import { render, screen } from '@testing-library/react';
import { CirculationStatusCard } from './circulation-status-card';
import { useCirculationsByCorrespondence } from '@/hooks/use-circulation';
// Mock hook สำหรับ useCirculationsByCorrespondence
vi.mock('@/hooks/use-circulation', () => ({
useCirculationsByCorrespondence: vi.fn(),
}));
describe('CirculationStatusCard Component', () => {
const correspondencePublicId = '019505a1-7c3e-7000-8000-abc123def456';
beforeEach(() => {
vi.clearAllMocks();
});
it('ควรแสดง loading state เมื่อกำลังโหลดข้อมูล', () => {
vi.mocked(useCirculationsByCorrespondence).mockReturnValue({
data: undefined,
isLoading: true,
} as any);
render(<CirculationStatusCard correspondencePublicId={correspondencePublicId} />);
expect(screen.getByText('Loading...')).toBeInTheDocument();
});
it('ควรแสดง empty state เมื่อไม่มีข้อมูล circulation', () => {
vi.mocked(useCirculationsByCorrespondence).mockReturnValue({
data: [],
isLoading: false,
} as any);
render(<CirculationStatusCard correspondencePublicId={correspondencePublicId} />);
expect(screen.getByText('No circulations yet')).toBeInTheDocument();
expect(screen.getByText('New Circulation')).toBeInTheDocument();
});
it('ควรแสดงรายการ circulation อย่างถูกต้องเมื่อโหลดสำเร็จ', () => {
const mockData = [
{
publicId: '019505a1-7c3e-7000-8000-abc123defaaa',
circulationNo: 'CIRC-2026-001',
subject: 'Circulation Subject A',
statusCode: 'OPEN',
routings: [
{
id: 1,
status: 'COMPLETED',
completedAt: '2026-06-13T00:00:00.000Z',
assignee: {
userId: 101,
username: 'john_doe',
firstName: 'John',
lastName: 'Doe',
},
},
{
id: 2,
status: 'PENDING',
assignee: {
userId: 102,
username: 'jane_smith',
firstName: 'Jane',
lastName: 'Smith',
},
},
],
},
];
vi.mocked(useCirculationsByCorrespondence).mockReturnValue({
data: mockData,
isLoading: false,
} as any);
render(<CirculationStatusCard correspondencePublicId={correspondencePublicId} />);
expect(screen.getByText('CIRC-2026-001')).toBeInTheDocument();
expect(screen.getByText('Circulation Subject A')).toBeInTheDocument();
expect(screen.getByText('OPEN')).toBeInTheDocument();
expect(screen.getByText('John Doe')).toBeInTheDocument();
expect(screen.getByText('Jane Smith')).toBeInTheDocument();
expect(screen.getByText('13 Jun')).toBeInTheDocument();
});
it('ควรแสดงข้อความ +X more assignees เมื่อมีผู้รับมากกว่า 3 คน', () => {
const mockData = [
{
publicId: '019505a1-7c3e-7000-8000-abc123defbbb',
circulationNo: 'CIRC-2026-002',
subject: 'Circulation Subject B',
statusCode: 'COMPLETED',
routings: [
{ id: 1, status: 'COMPLETED', assignee: { username: 'u1' } },
{ id: 2, status: 'COMPLETED', assignee: { username: 'u2' } },
{ id: 3, status: 'COMPLETED', assignee: { username: 'u3' } },
{ id: 4, status: 'COMPLETED', assignee: { username: 'u4' } },
{ id: 5, status: 'PENDING', assignee: { username: 'u5' } },
],
},
];
vi.mocked(useCirculationsByCorrespondence).mockReturnValue({
data: mockData,
isLoading: false,
} as any);
render(<CirculationStatusCard correspondencePublicId={correspondencePublicId} />);
expect(screen.getByText('+2 more assignees')).toBeInTheDocument();
});
});
@@ -0,0 +1,157 @@
// File: frontend/components/correspondences/tag-manager.test.tsx
// Change Log:
// - 2026-06-13: Initial creation - test coverage for tag-manager component
import { describe, it, expect, vi, beforeEach } from 'vitest';
import { render, screen, fireEvent, waitFor } from '@testing-library/react';
import { TagManager } from './tag-manager';
import { useCorrespondenceTags, useAddTag, useRemoveTag } from '@/hooks/use-correspondence';
import { useQuery } from '@tanstack/react-query';
import { masterDataService } from '@/lib/services/master-data.service';
// Mock React Query and hook implementations
vi.mock('@/hooks/use-correspondence', () => ({
useCorrespondenceTags: vi.fn(),
useAddTag: vi.fn(),
useRemoveTag: vi.fn(),
}));
vi.mock('@tanstack/react-query', () => ({
useQuery: vi.fn(),
}));
vi.mock('@/lib/services/master-data.service', () => ({
masterDataService: {
getTags: vi.fn(),
},
}));
describe('TagManager Component', () => {
const correspondenceUuid = '019505a1-7c3e-7000-8000-abc123def456';
const mockAddMutate = vi.fn();
const mockRemoveMutate = vi.fn();
beforeEach(() => {
vi.clearAllMocks();
vi.mocked(useQuery).mockReturnValue({
data: undefined,
isLoading: false,
} as any);
vi.mocked(useAddTag).mockReturnValue({
mutate: mockAddMutate,
isPending: false,
} as any);
vi.mocked(useRemoveTag).mockReturnValue({
mutate: mockRemoveMutate,
isPending: false,
} as any);
});
it('ควรแสดง loading state เมื่อกำลังโหลดข้อมูล tag', () => {
vi.mocked(useCorrespondenceTags).mockReturnValue({
data: undefined,
isLoading: true,
} as any);
render(<TagManager uuid={correspondenceUuid} canEdit={false} />);
expect(screen.getByText('Loading tags...')).toBeInTheDocument();
});
it('ควรแสดง empty state เมื่อไม่มี tag ถูกมอบหมาย', () => {
vi.mocked(useCorrespondenceTags).mockReturnValue({
data: [],
isLoading: false,
} as any);
render(<TagManager uuid={correspondenceUuid} canEdit={false} />);
expect(screen.getByText('No tags assigned')).toBeInTheDocument();
});
it('ควรแสดงรายการ tags ของเอกสารอย่างถูกต้อง', () => {
const mockTags = [
{ publicId: '019505a1-7c3e-7000-8000-tag111111111', tagName: 'Critical', colorCode: '#ff0000' },
{ publicId: '019505a1-7c3e-7000-8000-tag222222222', tagName: 'Draft', colorCode: 'default' },
];
vi.mocked(useCorrespondenceTags).mockReturnValue({
data: mockTags,
isLoading: false,
} as any);
render(<TagManager uuid={correspondenceUuid} canEdit={false} />);
expect(screen.getByText('Critical')).toBeInTheDocument();
expect(screen.getByText('Draft')).toBeInTheDocument();
});
it('ควรเรียก remove mutation เมื่อคลิกปุ่มลบ tag และมีสิทธิ์แก้ไข', () => {
const mockTags = [
{ publicId: '019505a1-7c3e-7000-8000-tag111111111', tagName: 'Critical', colorCode: '#ff0000' },
];
vi.mocked(useCorrespondenceTags).mockReturnValue({
data: mockTags,
isLoading: false,
} as any);
render(<TagManager uuid={correspondenceUuid} canEdit={true} />);
const removeBtn = screen.getAllByRole('button')[0];
fireEvent.click(removeBtn);
expect(mockRemoveMutate).toHaveBeenCalledWith({
uuid: correspondenceUuid,
tagId: '019505a1-7c3e-7000-8000-tag111111111',
});
});
it('ควรเปิดส่วนเลือก tag และแสดง tag ที่พร้อมให้เพิ่มเมื่อคลิก Add Tag', async () => {
const mockAssigned = [
{ publicId: '019505a1-7c3e-7000-8000-tag111111111', tagName: 'Critical', colorCode: '#ff0000' },
];
const mockAllTags = [
{ publicId: '019505a1-7c3e-7000-8000-tag111111111', tagName: 'Critical', colorCode: '#ff0000' },
{ publicId: '019505a1-7c3e-7000-8000-tag222222222', tagName: 'Draft', colorCode: '#00ff00' },
{ publicId: '019505a1-7c3e-7000-8000-tag333333333', tagName: 'Pending Review', colorCode: '#0000ff' },
];
vi.mocked(useCorrespondenceTags).mockReturnValue({
data: mockAssigned,
isLoading: false,
} as any);
vi.mocked(useQuery).mockReturnValue({
data: mockAllTags,
isLoading: false,
} as any);
render(<TagManager uuid={correspondenceUuid} canEdit={true} />);
const addTagBtn = screen.getByText('Add Tag');
fireEvent.click(addTagBtn);
await waitFor(() => {
expect(screen.getByText('Draft')).toBeInTheDocument();
expect(screen.getByText('Pending Review')).toBeInTheDocument();
});
// Tag 'Critical' มีอยู่แล้ว จึงไม่ควรปรากฏในส่วน list ที่พร้อมเพิ่ม
const listButtons = screen.getAllByRole('button');
const hasCriticalInList = listButtons.some(btn => btn.textContent === 'Critical');
expect(hasCriticalInList).toBe(false);
// คลิกเพื่อเพิ่ม tag 'Draft'
const draftBtn = screen.getByText('Draft');
fireEvent.click(draftBtn);
expect(mockAddMutate).toHaveBeenCalledWith({
uuid: correspondenceUuid,
tagId: '019505a1-7c3e-7000-8000-tag222222222',
});
});
});