Files
portal/app/tests/Frontend/AutopodborView.spec.ts
T
Дмитрий 4387333118 feat(Конкурентное поле): рабочее место конкуренты→источники→проекты (поверх автоподбора)
Фича «Конкурентное поле» на dev до уровня прототипа 2026-06-29-konkurentnoe-pole-proto.html.

Данные: box (proposal|field) на competitors+sources; phone_type city/mobile/tollfree рядом
с phone_kind (вариант C). 3 миграции, дефолты тарифов 300/50.

API (AutopodborController): GET /field (+счётчики), GET /proposals, PATCH/DELETE competitors
и sources с гвардами активного проекта, переключение box, POST /competitors/manual (+directory_urls),
competitor(id) обогащён box+project-статусом; projectStatus отдаёт limit/delivered/days/regions.
Смена источника проекта = PATCH /api/projects/{id} (реальный гвард слепка §14.10).

Фронт: FieldWorkspaceScreen/FieldCompetitorScreen/FieldProposalsScreen/FieldManualCompetitorScreen
+ field-shared.css (Forest) + AutopodborServicesPanel в Биллинге. Дословно по прототипу: подзаголовки,
баннер предложений, баннер правил времени 18:00 МСК, Справочник 2ГИС·Яндекс, статус проекта
5/день·заявки, окна сбора с ценами 300/50 + «что известно», полные формы. Пункт меню «Конкурентное поле».

Тесты: backend автоподбор 80/80, фронт автоподбор 49/49. Движок шага 2 = заглушка FakeCompetitorAgent.
OmegaDemoFieldSeeder — только для визуальной проверки (НЕ на прод).

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-30 04:18:46 +03:00

41 lines
1.7 KiB
TypeScript

import { describe, it, expect, beforeEach, vi } from 'vitest';
import { mount } from '@vue/test-utils';
import { createPinia, setActivePinia } from 'pinia';
import { createVuetify } from 'vuetify';
vi.mock('../../resources/js/api/autopodbor');
import * as api from '../../resources/js/api/autopodbor';
import AutopodborView from '../../resources/js/views/autopodbor/AutopodborView.vue';
const vuetify = createVuetify();
function mountView() {
return mount(AutopodborView, { global: { plugins: [vuetify] } });
}
describe('AutopodborView', () => {
beforeEach(() => {
setActivePinia(createPinia());
vi.clearAllMocks();
(api.fetchState as any).mockResolvedValue({ enabled: true, prices: { search: '500', study: '300' }, runs: [] });
(api.fetchField as any).mockResolvedValue([]);
});
it('по умолчанию показывает рабочее место «Конкурентное поле»', async () => {
const w = mountView();
await new Promise((r) => setTimeout(r, 0));
expect(w.text()).toContain('Конкурентное поле');
expect(w.text()).toContain('Собрать конкурентов');
});
it('«Собрать конкурентов для меня» открывает окно сбора', async () => {
const w = mountView();
await new Promise((r) => setTimeout(r, 0));
const btn = w.findAll('button').find((b) => b.text().includes('Собрать конкурентов для меня'));
expect(btn).toBeTruthy();
await btn!.trigger('click');
// открылось модальное окно сбора с правилами заполнения
expect(w.text()).toContain('Как заполнить, чтобы результат был точным');
});
});