Files
Nick Shirokov 91b39b758b test(web-test): M4.B+G — subordinate-nav + platform dialogs в 12-formstate
Расширены тесты getFormState: проверка ветвей navigation[] и
platformDialogs[] возвращаемой структуры.

- subordinate-nav: форма элемента Контрагент → state.navigation содержит
  «Основное» (active) и «Контактные лица» (подчинённый каталог).
- platform-dialogs: открытый через hamburger «О программе…» виден в
  state.platformDialogs[{type:'about'}].
- platform-dialog-close: closeForm закрывает платформенный диалог,
  массив становится пустым.

Покрыто 3 P2-кейса coverage matrix (12-formstate/subordinate-nav,
platform-dialogs, platform-dialog-close). Полный регресс 14/14 зелёный.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-11 16:30:51 +03:00

109 lines
6.0 KiB
JavaScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
export const name = 'getFormState: базовая структура — fields, buttons, tables, openForms';
export const tags = ['formstate', 'smoke'];
export const timeout = 60000;
export default async function({ navigateSection, openCommand, clickElement, closeForm, getFormState, getPage, assert, step, log }) {
await step('basic: getFormState на форме списка возвращает таблицу и команды', async () => {
await navigateSection('Склад');
const s = await openCommand('Контрагенты');
log(`form=${s.form} formCount=${s.formCount} tables=${s.tables?.length} buttons=${s.buttons?.length}`);
assert.ok(s.form != null, 'state.form задан');
assert.equal(s.formCount, 1, 'Открыта одна форма');
assert.ok(Array.isArray(s.openForms) && s.openForms.length === 1, 'openForms — массив с одной записью');
assert.ok(s.tables?.length >= 1, 'На форме списка есть таблица');
assert.ok(s.tables[0].columns?.length >= 2, 'У таблицы есть колонки');
assert.ok(s.buttons?.length >= 1, 'На форме есть кнопки');
await closeForm();
});
await step('basic: getFormState на форме элемента возвращает fields с label и value', async () => {
await navigateSection('Склад');
await openCommand('Контрагенты');
await clickElement('ООО Север', { dblclick: true });
const s = await getFormState();
log(`fields count=${s.fields?.length}`);
assert.ok(s.fields?.length >= 1, 'На форме элемента есть поля');
const named = s.fields.find(f => f.name === 'Наименование');
log(`Наименование: label='${named?.label}' value='${named?.value}'`);
assert.ok(named, 'Должно быть поле Наименование');
assert.equal(named.value, 'ООО Север', 'value поля Наименование');
assert.ok(named.label, 'У поля есть label');
await closeForm();
});
await step('modal: форма выбора Контрагентов открыта как модальная', async () => {
await navigateSection('Склад');
await openCommand('Приходная накладная');
await clickElement('Создать');
const page = await getPage();
// Найти input Контрагент и фокус, затем F4 → откроется модальная форма выбора
const focused = await page.evaluate(`(() => {
const inputs = [...document.querySelectorAll('input')];
const target = inputs.find(i => /Контрагент/i.test(i.id || '') && i.offsetWidth > 0);
if (target) { target.focus(); return target.id; }
return null;
})()`);
log(`focused input id=${focused}`);
await page.keyboard.press('F4');
await page.waitForTimeout(1500);
const s = await getFormState();
log(`after F4: form=${s.form} formCount=${s.formCount} modal=${s.modal}`);
assert.equal(s.modal, true, 'state.modal=true для модальной формы выбора');
assert.ok(s.formCount >= 2, 'formCount >= 2 (родитель + модальная)');
await closeForm();
await closeForm({ save: false });
});
await step('tabs: на форме элемента Номенклатуры присутствует tabs[]', async () => {
await navigateSection('Склад');
await openCommand('Номенклатура');
await clickElement('Товары', { dblclick: true });
await clickElement('Товар 01', { dblclick: true });
const s = await getFormState();
log(`tabs: ${JSON.stringify(s.tabs)}`);
assert.ok(Array.isArray(s.tabs), 'state.tabs должен быть массивом');
assert.ok(s.tabs.length >= 2, `На форме Номенклатуры >= 2 табов (got ${s.tabs.length})`);
await closeForm();
});
await step('subordinate-nav: форма элемента Контрагент возвращает state.navigation с КонтактнымиЛицами', async () => {
await navigateSection('Склад');
await openCommand('Контрагенты');
await clickElement('ООО Север', { dblclick: true });
const s = await getFormState();
log(`navigation: ${JSON.stringify(s.navigation)}`);
assert.ok(Array.isArray(s.navigation), 'state.navigation — массив');
assert.ok(s.navigation.length >= 2, 'минимум Основное + один подчинённый');
const main = s.navigation.find(n => n.active);
assert.ok(main && main.name === 'Основное', 'активная ссылка — Основное');
const sub = s.navigation.find(n => /Контактные/.test(n.name));
assert.ok(sub, 'есть ссылка на Контактные лица');
await closeForm();
});
await step('platform-dialogs: открытый «О программе» виден в state.platformDialogs', async () => {
const page = await getPage();
await page.click('#captionbarMore');
await page.waitForTimeout(800);
await page.getByText('О программе...', { exact: true }).click();
await page.waitForTimeout(1500);
const s = await getFormState();
log(`platformDialogs: ${JSON.stringify(s.platformDialogs)}`);
assert.ok(Array.isArray(s.platformDialogs) && s.platformDialogs.length === 1,
'state.platformDialogs — массив с одним элементом');
assert.equal(s.platformDialogs[0].type, 'about', 'type=about');
assert.equal(s.platformDialogs[0].title, 'О программе', 'title');
});
await step('platform-dialog-close: closeForm закрывает платформенный диалог', async () => {
// About остался открыт с предыдущего шага
await closeForm();
const s = await getFormState();
log(`platformDialogs after closeForm: ${s.platformDialogs?.length || 0}`);
assert.ok(!s.platformDialogs?.length, 'после closeForm нет platformDialogs');
});
}