test: add comprehensive test suite for components, composables, and pages

Add 16 new test files covering:
- Composables: useBackend, useMeta, useDataJson
- Type classes: QueryResult, ResumeContent
- UI components: BadgeList, BadgeListCard
- Navbar components: LanguageSwitcher, ThemeSwitcher
- App components: AppNavbar, AppFooter
- VocalSynth components: Projects, Tools
- Pages: contact, resume, [...slug]

Tests focus on pure logic, interfaces, and component rendering where
possible, avoiding complex mocking of Nuxt auto-imported composables.

Total: 174 tests across 17 test files (including existing useApi tests).
This commit is contained in:
2026-02-04 20:14:57 +01:00
parent e6a268bafd
commit 70e4ce8b4b
16 changed files with 2045 additions and 0 deletions

View File

@@ -0,0 +1,171 @@
import { describe, it, expect, vi } from 'vitest';
import { mountSuspended } from '@nuxt/test-utils/runtime';
import Projects from './Projects.vue';
import type { VocalSynthPage } from '~/types/vocal-synth';
// Mock $t function
vi.stubGlobal('$t', (key: string) => {
const translations: Record<string, string> = {
'pages.vocal-synthesis.projects': 'Projects',
};
return translations[key] || key;
});
describe('VocalSynth Projects', () => {
describe('external URL detection logic', () => {
const external = (url: string) => url.startsWith('http');
it('should return true for http URLs', () => {
expect(external('http://example.com')).toBe(true);
});
it('should return true for https URLs', () => {
expect(external('https://example.com')).toBe(true);
});
it('should return false for relative URLs', () => {
expect(external('/keine-tashi')).toBe(false);
});
it('should return false for paths without protocol', () => {
expect(external('/vocal-synthesis/project')).toBe(false);
});
});
describe('component rendering', () => {
it('should render the component', async () => {
const pageData: VocalSynthPage = {
projects: [],
tools: [],
};
const wrapper = await mountSuspended(Projects, {
global: {
provide: {
pageData,
},
},
});
expect(wrapper.exists()).toBe(true);
});
it('should display projects title', async () => {
const pageData: VocalSynthPage = {
projects: [],
tools: [],
};
const wrapper = await mountSuspended(Projects, {
global: {
provide: {
pageData,
},
},
});
expect(wrapper.text()).toContain('Projects');
});
it('should render projects from injected data', async () => {
const pageData: VocalSynthPage = {
projects: [
{
title: 'Test Project',
icon: 'mdi:music',
description: 'A test vocal synthesis project',
link: '/test-project',
},
],
tools: [],
};
const wrapper = await mountSuspended(Projects, {
global: {
provide: {
pageData,
},
},
});
expect(wrapper.text()).toContain('Test Project');
expect(wrapper.text()).toContain('A test vocal synthesis project');
});
it('should render multiple projects', async () => {
const pageData: VocalSynthPage = {
projects: [
{
title: 'Project One',
icon: 'mdi:music',
description: 'First project',
link: '/project-one',
},
{
title: 'Project Two',
icon: 'mdi:microphone',
description: 'Second project',
link: 'https://example.com/project-two',
},
],
tools: [],
};
const wrapper = await mountSuspended(Projects, {
global: {
provide: {
pageData,
},
},
});
expect(wrapper.text()).toContain('Project One');
expect(wrapper.text()).toContain('Project Two');
expect(wrapper.text()).toContain('First project');
expect(wrapper.text()).toContain('Second project');
});
it('should render project icons', async () => {
const pageData: VocalSynthPage = {
projects: [
{
title: 'Project with Icon',
icon: 'mdi:music-note',
description: 'Has an icon',
link: '/project',
},
],
tools: [],
};
const wrapper = await mountSuspended(Projects, {
global: {
provide: {
pageData,
},
},
});
// Icon container should exist
expect(wrapper.find('.min-w-13').exists()).toBe(true);
});
it('should handle empty projects array', async () => {
const pageData: VocalSynthPage = {
projects: [],
tools: [],
};
const wrapper = await mountSuspended(Projects, {
global: {
provide: {
pageData,
},
},
});
expect(wrapper.exists()).toBe(true);
expect(wrapper.text()).toContain('Projects');
});
});
});