4387333118
Фича «Конкурентное поле» на 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>
41 lines
1.7 KiB
TypeScript
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('Как заполнить, чтобы результат был точным');
|
|
});
|
|
});
|