From dab122c16616dbea7324d055240c24fd1a6bec4f Mon Sep 17 00:00:00 2001 From: Nick Shirokov Date: Fri, 12 Jun 2026 19:52:19 +0300 Subject: [PATCH] =?UTF-8?q?feat(form-decompile,form-compile):=20=D1=81?= =?UTF-8?q?=D0=B2=D0=BE=D0=B9=D1=81=D1=82=D0=B2=D0=B0=20=D0=BF=D0=BE=D0=BB?= =?UTF-8?q?=D1=8F=20DataSet=20=D0=B4=D0=B8=D0=BD=D1=81=D0=BF=D0=B8=D1=81?= =?UTF-8?q?=D0=BA=D0=B0=20=E2=80=94=20presentationExpression=20+=20appeara?= =?UTF-8?q?nce?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Обычное поле набора может нести: - (выражение представления поля, 30 форм) — строка; - (формат/оформление поля, ~9 форм) — dcscor:item SettingsParameterValue (тот же блок, что в условном оформлении: параметр→значение с типизацией). DSL: settings.fields[].presentationExpression (строка) + fields[].appearance (объект {параметр:значение}). Декомпилятор: захват presentationExpression + appearance через существующий Get-SettingsAppearance. Компилятор (ps1+py): presentationExpression перед valueType, appearance после valueType (порядок исходника, подтверждён корпус-сигнатурами); appearance переиспользует Emit-AppearanceValue/emit_appearance_value. Выборка 36 форм с field pres/appearance: match 33/36, 0 потерь pres/appearance (остаток 3 формы — несвязанные нюансы пустого value параметра / пустого LocalStringType). Кейс dynamic-list-form (+явное поле Code с presentationExpression+appearance Формат/ЦветТекста) сертифицирован загрузкой в 1С. Регресс 43/43, ps1==py (общий снэпшот на обоих рантаймах). Остаток field-свойств (BACKLOG): useRestriction/attributeUseRestriction/inputParameters/ order на обычном + 2 раскрытых нюанса (пустой xs:string value vs nil; пустой LocalStringType self-closing vs пара). Co-Authored-By: Claude Opus 4.8 (1M context) --- .../form-compile/scripts/form-compile.ps1 | 10 ++++++++- .../form-compile/scripts/form-compile.py | 11 +++++++++- .../form-decompile/scripts/form-decompile.ps1 | 7 ++++++- docs/form-dsl-spec.md | 2 +- .../cases/form-compile/dynamic-list-form.json | 4 ++++ .../Товары/Forms/ФормаСписка/Ext/Form.xml | 21 +++++++++++++++++++ 6 files changed, 51 insertions(+), 4 deletions(-) diff --git a/.claude/skills/form-compile/scripts/form-compile.ps1 b/.claude/skills/form-compile/scripts/form-compile.ps1 index 5d9aa810..330e57a4 100644 --- a/.claude/skills/form-compile/scripts/form-compile.ps1 +++ b/.claude/skills/form-compile/scripts/form-compile.ps1 @@ -1,4 +1,4 @@ -# form-compile v1.140 — Compile 1C managed form from JSON or object metadata +# form-compile v1.141 — Compile 1C managed form from JSON or object metadata # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills param( [string]$JsonPath, @@ -5626,8 +5626,16 @@ function Emit-Attributes { Emit-MLItems -val $fld.title -indent "$si`t`t" X "$si`t" } + # presentationExpression поля — перед valueType (порядок исходника) + if ($fld.presentationExpression) { X "$si`t$(Esc-Xml "$($fld.presentationExpression)")" } # valueType поля набора (тип значения; вычисляемые/кастомные поля) if ($fld.valueType) { Emit-DLValueType -typeStr "$($fld.valueType)" -indent "$si`t" } + # appearance поля (формат/оформление) — после valueType (порядок исходника) + if ($fld.appearance) { + X "$si`t" + foreach ($prop in $fld.appearance.PSObject.Properties) { Emit-AppearanceValue -key $prop.Name -val $prop.Value -indent "$si`t`t" } + X "$si`t" + } X "$si" } } diff --git a/.claude/skills/form-compile/scripts/form-compile.py b/.claude/skills/form-compile/scripts/form-compile.py index 8d42caf0..2a5ef656 100644 --- a/.claude/skills/form-compile/scripts/form-compile.py +++ b/.claude/skills/form-compile/scripts/form-compile.py @@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# form-compile v1.140 — Compile 1C managed form from JSON or object metadata +# form-compile v1.141 — Compile 1C managed form from JSON or object metadata # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills import argparse import copy @@ -5373,9 +5373,18 @@ def emit_attributes(lines, attrs, indent, conditional_appearance=None): lines.append(f'{si}\t') emit_ml_items(lines, f'{si}\t\t', fld['title']) lines.append(f'{si}\t') + # presentationExpression поля — перед valueType (порядок исходника) + if fld.get('presentationExpression'): + lines.append(f'{si}\t{esc_xml(str(fld["presentationExpression"]))}') # valueType поля набора (тип значения; вычисляемые/кастомные поля) if fld.get('valueType'): emit_dl_value_type(lines, fld['valueType'], f'{si}\t') + # appearance поля (формат/оформление) — после valueType (порядок исходника) + if fld.get('appearance'): + lines.append(f'{si}\t') + for ak, av in fld['appearance'].items(): + emit_appearance_value(lines, ak, av, f'{si}\t\t') + lines.append(f'{si}\t') lines.append(f'{si}') # Вычисляемые поля DataSet () — после Field*, до Parameter*. emit_calc_fields(lines, s.get('calculatedFields'), si) diff --git a/.claude/skills/form-decompile/scripts/form-decompile.ps1 b/.claude/skills/form-decompile/scripts/form-decompile.ps1 index 38decacb..2ae332ec 100644 --- a/.claude/skills/form-decompile/scripts/form-decompile.ps1 +++ b/.claude/skills/form-decompile/scripts/form-decompile.ps1 @@ -1,4 +1,4 @@ -# form-decompile v0.113 — Decompile 1C managed Form.xml to JSON DSL (draft) +# form-decompile v0.114 — Decompile 1C managed Form.xml to JSON DSL (draft) # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills # ВНИМАНИЕ: раундтрип не гарантируется. Навык исключён из авто-использования моделью. param( @@ -2614,6 +2614,11 @@ if ($attrsNode) { # valueType поля набора (тип значения; вычисляемые/кастомные поля). Грамматика типа. $fvt = $fn.SelectSingleNode("dcssch:valueType", $ns) if ($fvt) { $fvtVal = Decompile-Type $fvt; if ($fvtVal) { $fo['valueType'] = $fvtVal } } + # presentationExpression (выражение представления поля) + appearance (формат/оформление поля) + $fpe = Get-Child $fn 'presentationExpression' + if ($null -ne $fpe -and $fpe -ne '') { $fo['presentationExpression'] = $fpe } + $fappNode = $fn.SelectSingleNode("dcssch:appearance", $ns) + if ($fappNode) { $fap = Get-SettingsAppearance $fappNode; if ($fap -and $fap.Count -gt 0) { $fo['appearance'] = $fap } } [void]$fields.Add($fo) } $so['fields'] = @($fields) diff --git a/docs/form-dsl-spec.md b/docs/form-dsl-spec.md index 9eba688e..85194cbd 100644 --- a/docs/form-dsl-spec.md +++ b/docs/form-dsl-spec.md @@ -940,7 +940,7 @@ Forgiving-синонимы типа: XML-имя (`SpreadSheetDocumentField`) и | `query` | string | Текст запроса (`ManualQuery=true`). Поддерживает `@file.sql` (путь относительно JSON) | | `dynamicDataRead` | bool | Динамическое считывание. **Умолчание `true`** — указывать только для отключения (`false`) | | `autoFillAvailableFields` | bool | Автозаполнение доступных полей (``). **Умолчание `true`** — указывать только для отключения (`false`; тогда поля берутся из явного запроса, не авто). Эмитится первым в `` | -| `fields` | array | Явные поля набора (редко): `{ field, dataPath?, title?, valueType?, nested? }` — для переопределения заголовка/типа. `valueType` — тип значения поля (грамматика типа). `nested: true` помечает поле-вложенный набор (`DataSetFieldNestedDataSet` = реквизит табличной части объекта; дефолт — `DataSetFieldField`). Обычно поля выводятся из запроса автоматически | +| `fields` | array | Явные поля набора (редко): `{ field, dataPath?, title?, valueType?, presentationExpression?, appearance?, nested? }` — для переопределения свойств поля. `valueType` — тип значения (грамматика типа). `presentationExpression` — выражение представления поля (строка). `appearance` — оформление/формат поля: объект `{ Параметр: Значение }` (та же грамматика, что `appearance` условного оформления, напр. `{ "Формат": "ДЛ=9", "ЦветТекста": "web:Gray" }`). `nested: true` помечает поле-вложенный набор (`DataSetFieldNestedDataSet` = реквизит табличной части объекта; дефолт — `DataSetFieldField`). Обычно поля выводятся из запроса автоматически | | `calculatedFields` | array | Вычисляемые поля набора (см. ниже) | | `parameters` | array | Параметры схемы запроса (`DataCompositionSchemaParameter`) — см. ниже | | `order` | array | Сортировка списка (см. ниже) | diff --git a/tests/skills/cases/form-compile/dynamic-list-form.json b/tests/skills/cases/form-compile/dynamic-list-form.json index 9c0ea8af..9a645595 100644 --- a/tests/skills/cases/form-compile/dynamic-list-form.json +++ b/tests/skills/cases/form-compile/dynamic-list-form.json @@ -20,6 +20,10 @@ { "name": "Список", "type": "DynamicList", "useAlways": ["~Артикул", "Список.Code", "Description"], "settings": { "mainTable": "Catalog.Товары", "dynamicDataRead": true, "autoSaveUserSettings": false, "parameters": [ { "name": "ВыборТовара", "type": "CatalogRef.Товары", "valueListAllowed": true, "value": null } ], + "fields": [ + { "field": "Code", "title": "Код", "presentationExpression": "\"№\" + Code", + "appearance": { "Формат": "ДЛ=9", "ЦветТекста": "web:Gray" } } + ], "grouping": "Description > Code", "calculatedFields": [ "Метка [Метка строки]: string = Code + \" \" + Description #noFilter #noGroup", diff --git a/tests/skills/cases/form-compile/snapshots/dynamic-list-form/Catalogs/Товары/Forms/ФормаСписка/Ext/Form.xml b/tests/skills/cases/form-compile/snapshots/dynamic-list-form/Catalogs/Товары/Forms/ФормаСписка/Ext/Form.xml index 75e1435a..4cbcd771 100644 --- a/tests/skills/cases/form-compile/snapshots/dynamic-list-form/Catalogs/Товары/Forms/ФормаСписка/Ext/Form.xml +++ b/tests/skills/cases/form-compile/snapshots/dynamic-list-form/Catalogs/Товары/Forms/ФормаСписка/Ext/Form.xml @@ -100,6 +100,27 @@ false true + + Code + Code + + + ru + Код + + + "№" + Code + + + Формат + ДЛ=9 + + + ЦветТекста + web:Gray + + + Метка Code + " " + Description