mirror of
https://github.com/Nikolay-Shirokov/cc-1c-skills.git
synced 2026-06-11 08:24:57 +03:00
7de2689c18
Регресс-покрытие picture-колонок readTable на синтетическом стенде (без зависимости от реальных баз). В обработку ДеревоНоменклатуры: - булева колонка Картинка + PictureField (ValuesPicture=StdPicture.Favorites, loadTransparent) — иконка у позиций Цена>1000; - CheckBoxField Флаг на тот же булев (кросс-проверка состояния); - Selection-обработчик ДеревоВыбор — инверсия по двойному клику. 16-tree-form: обновлён deepEqual колонок (+Картинка +Флаг), добавлены шаги presence/кросс-проверка (pic:0 ⟺ флаг) и Selection-toggle. Полный регресс web-test зелёный. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
92 lines
6.5 KiB
JavaScript
92 lines
6.5 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.
|
||
|
||
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('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();
|
||
});
|
||
}
|