mirror of
https://github.com/Nikolay-Shirokov/cc-1c-skills.git
synced 2026-06-12 08:54:57 +03:00
fix(web-test): fillTableRow распознаёт переформатированные число/дату в choice-ячейке
fillChoiceCell определял «прижился ли paste» через normYo(after).includes(text), что ломалось на маск-инпутах: число/дата переформатируются (1234.56 → «1 234,56», группировка неразрывным пробелом, запятая) → includes давал false → ложный уход в F4, где у числа открывался калькулятор и залипал (no_selection_form). Заменил на поведенческий дискриминатор: появился EDD → ссылка (dropdown); инпут изменился на непустое без EDD → редактируемая ячейка (direct); инпут не изменился → НачалоВыбора → F4-форма. + страховка: если F4 открыл не форму выбора (калькулятор/календарь) — Escape и спасение значения. Также в EDD-ветке основного Tab-цикла убран слепой fallback items[0]: при отсутствии exact/includes-совпадения возвращается not_found с очисткой поля, а не подставляется произвольная первая запись автокомплита. Регресс: в стенд (дерево) добавлены choice-колонки Число/Дата и булево-поле-ввода; в 16-tree-form — шаги choice-number/choice-date/bool-input. Полный регресс: 22 passed. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -818,6 +818,14 @@ export const steps = [
|
||||
// Редактируемая строковая колонка: у поля есть кнопка выбора, но НачалоВыбора пустой
|
||||
// (F4 ничего не открывает), текст вводится напрямую — модель ячейки «Значение» Консоли запросов.
|
||||
{ name: 'РедактируемаяСтрока', type: 'String', title: 'Редактируемая строка' },
|
||||
// Редактируемые choice-ячейки Число и Дата (та же модель «Значение» КЗ): кнопка выбора +
|
||||
// пустой НачалоВыбора, текст вводится напрямую и ПЕРЕФОРМАТИРУЕТСЯ маск-инпутом
|
||||
// (1234.56 → «1 234,56»). Регресс-guard для fillChoiceCell — раньше includes-проверка
|
||||
// рвалась о переформатирование → ложное F4 → калькулятор.
|
||||
{ name: 'РедактируемоеЧисло', type: 'Number(15,2)', title: 'Редактируемое число' },
|
||||
{ name: 'РедактируемаяДата', type: 'date', title: 'Редактируемая дата' },
|
||||
// Булева колонка-флажок (отдельно от Картинка) — для fillTableRow toggle на дереве.
|
||||
{ name: 'Булево', type: 'Boolean', title: 'Булево' },
|
||||
]},
|
||||
// Список значений для программного выбора (ПоказатьВыборЭлемента).
|
||||
{ name: 'СписокТипов', type: 'ValueList' },
|
||||
@@ -841,6 +849,17 @@ export const steps = [
|
||||
// кнопка iCB есть, F4 ничего не открывает, текст редактируется напрямую (модель «Значение»).
|
||||
{ input: 'ДеревоРедактируемаяСтрока', path: 'Дерево.РедактируемаяСтрока', title: 'Редактируемая строка',
|
||||
choiceButton: true, on: ['StartChoice'], handlers: { StartChoice: 'ДеревоРедактируемаяСтрокаНачалоВыбора' } },
|
||||
// Редактируемые choice-ячейки Число/Дата: кнопка iCB + пустой НачалоВыбора → текст
|
||||
// редактируется напрямую, значение переформатируется маск-инпутом (модель «Значение» КЗ).
|
||||
{ input: 'ДеревоРедактируемоеЧисло', path: 'Дерево.РедактируемоеЧисло', title: 'Редактируемое число',
|
||||
choiceButton: true, on: ['StartChoice'], handlers: { StartChoice: 'ДеревоРедактируемоеЧислоНачалоВыбора' } },
|
||||
{ input: 'ДеревоРедактируемаяДата', path: 'Дерево.РедактируемаяДата', title: 'Редактируемая дата',
|
||||
choiceButton: true, on: ['StartChoice'], handlers: { StartChoice: 'ДеревоРедактируемаяДатаНачалоВыбора' } },
|
||||
// Булево как ПОЛЕ ВВОДА с кнопкой выбора (не флажок): в ячейке выбор Да/Нет —
|
||||
// fillTableRow идёт через dropdown-путь (как «Значение» типа Булево в Консоли
|
||||
// запросов), не toggle. Кнопка iCB + пустой НачалоВыбора — единая модель «Значение».
|
||||
{ input: 'ДеревоБулево', path: 'Дерево.Булево', title: 'Булево',
|
||||
choiceButton: true, on: ['StartChoice'], handlers: { StartChoice: 'ДеревоБулевоНачалоВыбора' } },
|
||||
]},
|
||||
],
|
||||
},
|
||||
@@ -926,6 +945,25 @@ export const steps = [
|
||||
\t// Текст вводится напрямую — модель ячейки «Значение» типовой Консоли запросов.
|
||||
\tСтандартнаяОбработка = Ложь;
|
||||
КонецПроцедуры
|
||||
|
||||
&НаКлиенте
|
||||
Процедура ДеревоРедактируемоеЧислоНачалоВыбора(Элемент, ДанныеВыбора, СтандартнаяОбработка)
|
||||
\t// Пустой обработчик — число редактируется напрямую (модель «Значение» КЗ).
|
||||
\tСтандартнаяОбработка = Ложь;
|
||||
КонецПроцедуры
|
||||
|
||||
&НаКлиенте
|
||||
Процедура ДеревоРедактируемаяДатаНачалоВыбора(Элемент, ДанныеВыбора, СтандартнаяОбработка)
|
||||
\t// Пустой обработчик — дата редактируется напрямую (модель «Значение» КЗ).
|
||||
\tСтандартнаяОбработка = Ложь;
|
||||
КонецПроцедуры
|
||||
|
||||
&НаКлиенте
|
||||
Процедура ДеревоБулевоНачалоВыбора(Элемент, ДанныеВыбора, СтандартнаяОбработка)
|
||||
\t// Пустой обработчик: кнопка выбора есть, F4 ничего не открывает; значение задаётся
|
||||
\t// штатным списком Да/Нет поля ввода булева (модель «Значение» типа Булево в КЗ).
|
||||
\tСтандартнаяОбработка = Ложь;
|
||||
КонецПроцедуры
|
||||
`,
|
||||
},
|
||||
|
||||
|
||||
@@ -14,6 +14,9 @@ export const timeout = 90000;
|
||||
// + дискриминатор choice-ячейки (fillChoiceCell): ДеревоРедактируемаяСтрока (кнопка iCB,
|
||||
// пустой НачалоВыбора, текст редактируется → method:'direct') vs ДеревоТипЗначения
|
||||
// (РедактированиеТекста=Ложь, текст отвергается → форма выбора, method:'choice').
|
||||
// + редактируемые choice-ячейки Число/Дата (РедактируемоеЧисло/РедактируемаяДата): маск-инпут
|
||||
// переформатирует значение (1234.56 → «1 234,56») — регресс на баг с ложным F4→калькулятор.
|
||||
// + булево как поле ввода (Булево, InputField, не флажок): выбор Да/Нет через dropdown-путь.
|
||||
|
||||
export default async function({ navigateLink, clickElement, closeForm, readTable, fillTableRow, assert, step, log }) {
|
||||
|
||||
@@ -27,7 +30,7 @@ export default async function({ navigateLink, clickElement, closeForm, readTable
|
||||
await step('read-roots: на верхнем уровне видны группы (Товары, Услуги, БольшойСписок)', async () => {
|
||||
const t = await readTable('Дерево');
|
||||
log(`columns=${t.columns?.join(',')} rows=${t.rows?.length}`);
|
||||
assert.deepEqual(t.columns, ['Номенклатура', 'Цена', 'Картинка', 'Флаг', 'Тип значения', 'Редактируемая строка'], 'колонки: Номенклатура + Цена + Картинка + Флаг + Тип значения + Редактируемая строка');
|
||||
assert.deepEqual(t.columns, ['Номенклатура', 'Цена', 'Картинка', 'Флаг', 'Тип значения', 'Редактируемая строка', 'Редактируемое число', 'Редактируемая дата', 'Булево'], 'колонки: Номенклатура + Цена + Картинка + Флаг + Тип значения + Редактируемая строка + Редактируемое число + Редактируемая дата + Булево');
|
||||
assert.equal(t.rows.length, 3, '3 корневые строки');
|
||||
const names = t.rows.map(r => r['Номенклатура']);
|
||||
assert.includes(names, 'Товары', 'есть Товары');
|
||||
@@ -119,6 +122,50 @@ export default async function({ navigateLink, clickElement, closeForm, readTable
|
||||
assert.equal(cell.error, 'not_found', 'error=not_found');
|
||||
});
|
||||
|
||||
await step('choice-number: редактируемая choice-ячейка Число — paste переформатируется, method:direct', async () => {
|
||||
// РедактируемоеЧисло — Number-колонка с кнопкой выбора (iCB) и пустым НачалоВыбора (модель
|
||||
// «Значение» КЗ). Маск-инпут ПЕРЕФОРМАТИРУЕТ «1234.56» → «1 234,56» (nbsp-разделитель, запятая).
|
||||
// Регресс на баг, где includes-проверка рвалась о переформатирование → ложное F4 → калькулятор.
|
||||
// Теперь дискриминатор поведенческий (инпут изменился + нет EDD → direct), формат не важен.
|
||||
const r = await fillTableRow({ 'Редактируемое число': '1234.56' }, { 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 (числовой маск-инпут, без F4/калькулятора)');
|
||||
const t = await readTable('Дерево');
|
||||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||||
// 1С web использует неразрывный пробел как разделитель разрядов — убираем все пробелы перед сравнением.
|
||||
assert.equal((tovar01['Редактируемое число'] || '').replace(/[\s\u00A0]/g, ''), '1234,56', 'число переформатировано в 1234,56');
|
||||
});
|
||||
|
||||
await step('choice-date: редактируемая choice-ячейка Дата — method:direct', async () => {
|
||||
// РедактируемаяДата — Date-колонка с кнопкой выбора и пустым НачалоВыбора. Та же модель «Значение».
|
||||
const r = await fillTableRow({ 'Редактируемая дата': '15.06.2025' }, { 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['Редактируемая дата'], '15.06.2025', 'дата записана');
|
||||
});
|
||||
|
||||
await step('bool-input: булева ячейка-поле-ввода (не флажок) заполняется выбором Да', async () => {
|
||||
// ДеревоБулево — InputField на булевом пути (НЕ CheckBoxField) с кнопкой выбора (iCB) и
|
||||
// пустым НачалоВыбора: в ячейке выбор Да/Нет, fillTableRow идёт через dropdown-путь, а не
|
||||
// toggle. Покрывает булево как поле ввода (модель «Значение» типа Булево в Консоли запросов).
|
||||
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');
|
||||
const t = await readTable('Дерево');
|
||||
const tovar01 = t.rows.find(row => row['Номенклатура'] === 'Товар 01');
|
||||
assert.equal(tovar01['Булево'], 'Да', 'Булево = Да');
|
||||
});
|
||||
|
||||
await step('picture: колонка-картинка (pic:0/\'\') + кросс-проверка чекбоксом', async () => {
|
||||
const t = await readTable('Дерево');
|
||||
const t15 = t.rows.find(r => r['Номенклатура'] === 'Товар 15'); // Цена 1500 > 1000 → иконка
|
||||
|
||||
Reference in New Issue
Block a user