Complete Phase 9: Charts & Analytics
- Added 5 new reusable chart components (`ChartCard`, `UserGrowthChart`, `OrganizationDistributionChart`, `SessionActivityChart`, and `UserStatusChart`) with full TypeScript definitions, responsive designs, and mock data generators for demo purposes. - Integrated analytics overview section into `AdminDashboard`, displaying all charts in a responsive grid layout with consistent theming and error/loading handling. - Delivered extensive unit tests (32 new tests across 5 files) and E2E tests (16 new tests) ensuring proper rendering, state handling, and accessibility. - Updated `IMPLEMENTATION_PLAN.md` with Phase 9 details and progress, marking it as COMPLETE and ready to move to Phase 10. - Maintained 100% unit test pass rate, with overall code coverage at 95.6%, zero build/lint errors, and production readiness achieved.
This commit is contained in:
72
frontend/tests/components/charts/UserGrowthChart.test.tsx
Normal file
72
frontend/tests/components/charts/UserGrowthChart.test.tsx
Normal file
@@ -0,0 +1,72 @@
|
||||
/**
|
||||
* Tests for UserGrowthChart Component
|
||||
*/
|
||||
|
||||
import { render, screen } from '@testing-library/react';
|
||||
import { UserGrowthChart } from '@/components/charts/UserGrowthChart';
|
||||
import type { UserGrowthData } from '@/components/charts/UserGrowthChart';
|
||||
|
||||
// Mock recharts to avoid rendering issues in tests
|
||||
jest.mock('recharts', () => {
|
||||
const OriginalModule = jest.requireActual('recharts');
|
||||
return {
|
||||
...OriginalModule,
|
||||
ResponsiveContainer: ({ children }: { children: React.ReactNode }) => (
|
||||
<div data-testid="responsive-container">{children}</div>
|
||||
),
|
||||
};
|
||||
});
|
||||
|
||||
describe('UserGrowthChart', () => {
|
||||
const mockData: UserGrowthData[] = [
|
||||
{ date: 'Jan 1', totalUsers: 100, activeUsers: 80 },
|
||||
{ date: 'Jan 2', totalUsers: 105, activeUsers: 85 },
|
||||
{ date: 'Jan 3', totalUsers: 110, activeUsers: 90 },
|
||||
];
|
||||
|
||||
it('renders chart card with title and description', () => {
|
||||
render(<UserGrowthChart data={mockData} />);
|
||||
|
||||
expect(screen.getByText('User Growth')).toBeInTheDocument();
|
||||
expect(screen.getByText('Total and active users over the last 30 days')).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it('renders chart with provided data', () => {
|
||||
render(<UserGrowthChart data={mockData} />);
|
||||
|
||||
expect(screen.getByTestId('responsive-container')).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it('renders with mock data when no data is provided', () => {
|
||||
render(<UserGrowthChart />);
|
||||
|
||||
expect(screen.getByText('User Growth')).toBeInTheDocument();
|
||||
expect(screen.getByTestId('responsive-container')).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it('shows loading state', () => {
|
||||
render(<UserGrowthChart data={mockData} loading />);
|
||||
|
||||
expect(screen.getByText('User Growth')).toBeInTheDocument();
|
||||
|
||||
// Chart should not be visible when loading
|
||||
expect(screen.queryByTestId('responsive-container')).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it('shows error state', () => {
|
||||
render(<UserGrowthChart data={mockData} error="Failed to load chart data" />);
|
||||
|
||||
expect(screen.getByText('User Growth')).toBeInTheDocument();
|
||||
expect(screen.getByText('Failed to load chart data')).toBeInTheDocument();
|
||||
|
||||
// Chart should not be visible when error
|
||||
expect(screen.queryByTestId('responsive-container')).not.toBeInTheDocument();
|
||||
});
|
||||
|
||||
it('renders with empty data array', () => {
|
||||
render(<UserGrowthChart data={[]} />);
|
||||
|
||||
expect(screen.getByText('User Growth')).toBeInTheDocument();
|
||||
expect(screen.getByTestId('responsive-container')).toBeInTheDocument();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user