import { describe, it, expect, beforeEach, vi } from 'vitest'; import { mount } from '@vue/test-utils'; import { createVuetify } from 'vuetify'; import axios from 'axios'; import AdminSupplierPricesView from '../../resources/js/views/admin/AdminSupplierPricesView.vue'; vi.mock('axios'); // Auto-импорт компонентов/директив Vuetify подхватывает vite-plugin-vuetify // из vitest.config.ts (см. AdminPricingTiersView.spec.ts). const vuetify = createVuetify(); const mockSuppliers = [ { id: 1, code: 'b1', name: 'B1 — Сайты и Звонки', cost_rub: '1.00', quality_score: '1.00', is_active: true }, { id: 2, code: 'b2', name: 'B2 — SMS', cost_rub: '1.50', quality_score: '1.00', is_active: true }, { id: 3, code: 'b3', name: 'B3 — SMS', cost_rub: '1.20', quality_score: '0.95', is_active: true }, ]; describe('AdminSupplierPricesView', () => { beforeEach(() => { // eslint-disable-next-line @typescript-eslint/no-explicit-any (axios.get as any).mockResolvedValue({ data: { data: mockSuppliers } }); // eslint-disable-next-line @typescript-eslint/no-explicit-any (axios.patch as any).mockResolvedValue({ data: { data: mockSuppliers[0] } }); }); it('renders 3 supplier rows', async () => { const wrapper = mount(AdminSupplierPricesView, { global: { plugins: [vuetify] } }); await new Promise((r) => setTimeout(r, 50)); expect(wrapper.text()).toContain('b1'); expect(wrapper.text()).toContain('b2'); expect(wrapper.text()).toContain('b3'); }); it('save() fires PATCH with cost_rub/quality_score/is_active', async () => { const wrapper = mount(AdminSupplierPricesView, { global: { plugins: [vuetify] } }); await new Promise((r) => setTimeout(r, 50)); // eslint-disable-next-line @typescript-eslint/no-explicit-any await (wrapper.vm as any).save({ id: 1, code: 'b1', name: '', cost_rub: '2.00', quality_score: '1.00', is_active: true, }); expect(axios.patch).toHaveBeenCalledWith('/api/admin/suppliers/1', { cost_rub: '2.00', quality_score: '1.00', is_active: true, }); }); it('renders quality_score, cost_rub as editable text-fields', async () => { const wrapper = mount(AdminSupplierPricesView, { global: { plugins: [vuetify] } }); await new Promise((r) => setTimeout(r, 50)); const inputs = wrapper.findAll('input[type="number"]'); expect(inputs.length).toBeGreaterThanOrEqual(6); }); it('each input/switch has explicit aria-label combining supplier name + field role', async () => { const wrapper = mount(AdminSupplierPricesView, { global: { plugins: [vuetify] } }); await new Promise((r) => setTimeout(r, 50)); // 3 suppliers × 3 fields = 9 controls const expectedLabels = [ 'Cost (₽) для B1 — Сайты и Звонки', 'Quality для B1 — Сайты и Звонки', 'Active для B1 — Сайты и Звонки', 'Cost (₽) для B2 — SMS', 'Quality для B2 — SMS', 'Active для B2 — SMS', 'Cost (₽) для B3 — SMS', 'Quality для B3 — SMS', 'Active для B3 — SMS', ]; for (const label of expectedLabels) { const node = wrapper.find(`[aria-label="${label}"]`); expect(node.exists(), `aria-label="${label}" not found`).toBe(true); } }); });