mirror of
https://github.com/Nikolay-Shirokov/cc-1c-skills.git
synced 2026-06-12 00:44:57 +03:00
ffb380187f
Диалог выбора типа матчил по подстроке и падал «multiple types match», даже когда точное совпадение присутствовало в выдаче (напр. поиск «Контрагент» давал «Банковская карта контрагента», «Договор с контрагентом», …, «Контрагент» — и движок ругался, хотя точная строка была видна). pickFromTypeDialog теперь предпочитает точное совпадение (resolveExact: единственный матч, либо единственная строка, равная искомому имени после нормализации регистра/ё) — кликает именно её и жмёт OK. Применяется и в scan-пути (мелкие списки), и после Ctrl+F (большие виртуальные списки). Добавлен ограниченный скролл-скан (PageDown ×3) на случай, когда точная строка чуть ниже первого окна. Ошибка неоднозначности остаётся, только если единственного точного совпадения действительно нет. Стенд: в СписокТипов добавлен подстрочный дубль «Дата документа» рядом с «Дата» для детерминированной проверки exact-match. Тест 16-tree-form покрывает scan-путь (выбирается точное «Дата»). Проверено: регресс web-test 22/22, живой E2E на типовой Консоли запросов (ссылочный тип через Ctrl+F + примитив без регресса). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
150 lines
12 KiB
JavaScript
150 lines
12 KiB
JavaScript
export const name = 'tree-form: FormDataTree edit (ДеревоНоменклатуры obrabotka)';
|
||
export const tags = ['tree', 'table', 'picture'];
|
||
export const timeout = 90000;
|
||
|
||
// ДеревоНоменклатуры obrabotka: реквизит формы Дерево типа ДеревоЗначений
|
||
// заполняется в ПриСозданииНаСервере рекурсивным обходом справочника Номенклатура.
|
||
// Колонки: Номенклатура (CatalogRef, readOnly), Цена (Number, editable),
|
||
// Картинка (PictureField на булев, ValuesPicture=StdPicture.Favorites, иконка у Цена>1000),
|
||
// Флаг (CheckBoxField на тот же булев — кросс-проверка). Selection-обработчик
|
||
// ДеревоВыбор инвертирует Картинка по двойному клику в колонке.
|
||
// Покрывает: 05-table/edit-form (fillTableRow method:'direct' на FormDataTree-колонке)
|
||
// + 08-hierarchy/tree-edit (expand узла + edit Цены внутри expanded группы)
|
||
// + readTable picture-колонки (pic:N/'') и Selection-toggle.
|
||
// + дискриминатор choice-ячейки (fillChoiceCell): ДеревоРедактируемаяСтрока (кнопка iCB,
|
||
// пустой НачалоВыбора, текст редактируется → method:'direct') vs ДеревоТипЗначения
|
||
// (РедактированиеТекста=Ложь, текст отвергается → форма выбора, method:'choice').
|
||
|
||
export default async function({ navigateLink, clickElement, closeForm, readTable, fillTableRow, assert, step, log }) {
|
||
|
||
await step('setup: открыть обработку ДеревоНоменклатуры', async () => {
|
||
const r = await navigateLink('Обработка.ДеревоНоменклатуры');
|
||
log(`form=${r.form} activeTab=${r.activeTab}`);
|
||
assert.equal(r.activeTab, 'Дерево номенклатуры', 'форма открыта');
|
||
assert.ok(r.tables?.some(t => t.name === 'Дерево'), 'таблица Дерево присутствует');
|
||
});
|
||
|
||
await step('read-roots: на верхнем уровне видны группы (Товары, Услуги, БольшойСписок)', async () => {
|
||
const t = await readTable('Дерево');
|
||
log(`columns=${t.columns?.join(',')} rows=${t.rows?.length}`);
|
||
assert.deepEqual(t.columns, ['Номенклатура', 'Цена', 'Картинка', 'Флаг', 'Тип значения', 'Редактируемая строка'], 'колонки: Номенклатура + Цена + Картинка + Флаг + Тип значения + Редактируемая строка');
|
||
assert.equal(t.rows.length, 3, '3 корневые строки');
|
||
const names = t.rows.map(r => r['Номенклатура']);
|
||
assert.includes(names, 'Товары', 'есть Товары');
|
||
assert.includes(names, 'Услуги', 'есть Услуги');
|
||
assert.includes(names, 'БольшойСписок', 'есть БольшойСписок');
|
||
assert.ok(t.rows.every(r => r._kind === 'group'), 'все корневые — group (есть expand-стрелка)');
|
||
});
|
||
|
||
await step('expand: clickElement({expand}) раскрывает Товары — 15 элементов', async () => {
|
||
const r = await clickElement('Товары', { expand: true });
|
||
log(`clicked: ${JSON.stringify(r.clicked)}`);
|
||
assert.equal(r.clicked?.toggled, true, 'expand toggled');
|
||
const t = await readTable('Дерево');
|
||
log(`after expand: total=${t.total}`);
|
||
assert.ok(t.total >= 16, `Товары + 15 элементов (got ${t.total})`);
|
||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||
assert.ok(tovar01, 'Товар 01 виден внутри Товары');
|
||
assert.equal(tovar01['Цена'], '100,00', 'исходная Цена 100,00 (из справочника)');
|
||
});
|
||
|
||
await step('tree-edit: fillTableRow меняет Цену в развёрнутой группе', async () => {
|
||
// row:1 — это Товар 01 (row:0 — Товары после expand). Используем index, т.к.
|
||
// fillTableRow{row:'Товар 01'} ловит SyntaxError в JS-эвале — TODO в bug list.
|
||
const r = await fillTableRow({ Цена: 1500 }, { row: 1 });
|
||
log(`filled: ${JSON.stringify(r.filled)}`);
|
||
assert.equal(r.filled?.length, 1, '1 поле заполнено');
|
||
assert.equal(r.filled[0].field, 'Цена', 'поле Цена');
|
||
assert.equal(r.filled[0].method, 'direct', 'method=direct (in-place edit)');
|
||
assert.equal(r.filled[0].ok, true, 'ok=true');
|
||
const t = await readTable('Дерево');
|
||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||
assert.ok(tovar01, 'Товар 01 виден');
|
||
// 1С web использует non-breaking space ( ) как разделитель разрядов
|
||
assert.equal(tovar01['Цена'], '1 500,00', 'Цена обновилась до 1 500,00');
|
||
});
|
||
|
||
await step('choice-direct: редактируемая choice-ячейка заполняется прямым вводом (method:direct)', async () => {
|
||
// ДеревоРедактируемаяСтрока — поле с кнопкой выбора (iCB), но пустым НачалоВыбора и
|
||
// РедактированиеТекста=Истина: текст ПРИЛИПАЕТ. fillChoiceCell определяет это поведенчески
|
||
// (paste прилип → stuck) и вводит напрямую, не уходя в форму. Модель ячейки «Значение»
|
||
// типовой Консоли запросов (была баг no_selection_form).
|
||
const r = await fillTableRow({ 'Редактируемая строка': 'привет' }, { row: 1 });
|
||
log(`filled: ${JSON.stringify(r.filled)}`);
|
||
const cell = r.filled?.find(f => /Редактируем/i.test(f.field));
|
||
assert.ok(cell, 'поле Редактируемая строка в результате');
|
||
assert.equal(cell.ok, true, 'ok=true');
|
||
assert.equal(cell.method, 'direct', 'method=direct (прямой ввод, форма не открывалась)');
|
||
const t = await readTable('Дерево');
|
||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||
assert.equal(tovar01['Редактируемая строка'], 'привет', 'значение введено напрямую');
|
||
});
|
||
|
||
await step('choice-cell: fillTableRow задаёт ТипЗначения через форму выбора (НачалоВыбора)', async () => {
|
||
// Колонка-строка с кнопкой выбора + обработчиком НачалоВыбора → СписокТипов.ПоказатьВыборЭлемента
|
||
// («Выбрать тип»), РедактированиеТекста=Ложь. Прямой ввод ОТВЕРГАется — fillChoiceCell видит
|
||
// stuck=false и открывает форму выбора, выбирая из списка (method:choice, не direct).
|
||
const r = await fillTableRow({ ТипЗначения: 'Число' }, { row: 1 });
|
||
log(`filled: ${JSON.stringify(r.filled)}`);
|
||
const cell = r.filled?.find(f => f.field === 'ТипЗначения');
|
||
assert.ok(cell, 'поле ТипЗначения в результате');
|
||
assert.equal(cell.ok, true, 'ok=true');
|
||
assert.equal(cell.method, 'choice', 'method=choice (выбор из списка, не direct)');
|
||
const t = await readTable('Дерево');
|
||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||
assert.equal(tovar01['Тип значения'], 'Число', 'ТипЗначения = Число');
|
||
});
|
||
|
||
await step('choice-exact: при подстрочной неоднозначности выбирается точное совпадение', async () => {
|
||
// СписокТипов содержит «Дата» и «Дата документа». Поиск «Дата» даёт 2 подстрочных
|
||
// совпадения — pickFromTypeDialog должен предпочесть ТОЧНОЕ «Дата», а не ругаться
|
||
// на неоднозначность и не выбрать «Дата документа» (Проблема 2 из bug-report).
|
||
const r = await fillTableRow({ ТипЗначения: 'Дата' }, { row: 1 });
|
||
const cell = r.filled?.find(f => f.field === 'ТипЗначения');
|
||
assert.ok(cell, 'поле ТипЗначения в результате');
|
||
assert.equal(cell.ok, true, 'ok=true (exact-match разрешил неоднозначность)');
|
||
assert.equal(cell.method, 'choice', 'method=choice');
|
||
const t = await readTable('Дерево');
|
||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||
assert.equal(tovar01['Тип значения'], 'Дата', 'выбрано точное «Дата», не «Дата документа»');
|
||
});
|
||
|
||
await step('choice-cell-negative: несуществующий тип → ok:false/not_found (форма не закрывается)', async () => {
|
||
// not_found гасит только диалог выбора типа (умный dismiss), исходная форма остаётся —
|
||
// следующие шаги (picture) это подтверждают.
|
||
const r = await fillTableRow({ ТипЗначения: 'Нетакоготипа' }, { row: 1 });
|
||
const cell = r.filled?.find(f => f.field === 'ТипЗначения');
|
||
assert.ok(cell, 'поле ТипЗначения в результате');
|
||
assert.equal(cell.ok, false, 'ok=false для несуществующего типа');
|
||
assert.equal(cell.error, 'not_found', 'error=not_found');
|
||
});
|
||
|
||
await step('picture: колонка-картинка (pic:0/\'\') + кросс-проверка чекбоксом', async () => {
|
||
const t = await readTable('Дерево');
|
||
const t15 = t.rows.find(r => r['Номенклатура'] === 'Товар 15'); // Цена 1500 > 1000 → иконка
|
||
const t02 = t.rows.find(r => r['Номенклатура'] === 'Товар 02'); // Цена 200 < 1000 → нет
|
||
assert.ok(t15 && t02, 'Товар 15 и Товар 02 видны');
|
||
assert.equal(t15['Картинка'], 'pic:0', 'Товар 15 — иконка показана (pic:0)');
|
||
assert.equal(t15['Флаг'], 'true', 'Товар 15 — флаг true');
|
||
assert.equal(t02['Картинка'], '', 'Товар 02 — иконки нет (пусто)');
|
||
assert.equal(t02['Флаг'], 'false', 'Товар 02 — флаг false');
|
||
// Инвариант: иконка есть тогда и только тогда, когда флаг true.
|
||
const items = t.rows.filter(r => (r['Номенклатура'] || '').startsWith('Товар '));
|
||
assert.ok(items.length >= 15, 'видны все 15 товаров');
|
||
assert.ok(items.every(r => (r['Картинка'] === 'pic:0') === (r['Флаг'] === 'true')),
|
||
'по всем товарам: картинка ⟺ флаг');
|
||
});
|
||
|
||
await step('picture-toggle: Selection инвертирует Картинка по двойному клику', async () => {
|
||
await clickElement({ row: { 'Номенклатура': 'Товар 02' }, column: 'Картинка' }, { dblclick: true });
|
||
const t = await readTable('Дерево');
|
||
const t02 = t.rows.find(r => r['Номенклатура'] === 'Товар 02');
|
||
assert.equal(t02['Картинка'], 'pic:0', 'после двойного клика иконка появилась');
|
||
assert.equal(t02['Флаг'], 'true', 'и флаг стал true');
|
||
});
|
||
|
||
await step('cleanup: закрыть форму', async () => {
|
||
await closeForm();
|
||
});
|
||
}
|