diff --git a/.claude/skills/form-compile/scripts/form-compile.ps1 b/.claude/skills/form-compile/scripts/form-compile.ps1 index 330e57a4..8c35aaff 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.141 — Compile 1C managed form from JSON or object metadata +# form-compile v1.142 — Compile 1C managed form from JSON or object metadata # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills param( [string]$JsonPath, @@ -2141,6 +2141,31 @@ function Emit-CalcFields { } } +# Ограничения использования поля/вычисляемого поля (useRestriction / attributeUseRestriction). +# Значение: объект {field?,condition?,group?,order?} | флаг-строка "#noField #noFilter #noGroup #noOrder" | массив. +function Get-RestrictList { + param($ur) + $out = @() + if (-not $ur) { return ,$out } + if ($ur -is [System.Management.Automation.PSCustomObject] -or $ur -is [hashtable]) { + foreach ($k in 'field','condition','group','order') { if ($ur.$k -eq $true) { $out += $k } } + } elseif ($ur -is [string]) { + foreach ($tok in ($ur -split '\s+')) { $t = $tok.Trim().TrimStart('#'); if ($t) { $out += $(if ($script:calcRestrictMap[$t]) { $script:calcRestrictMap[$t] } else { $t }) } } + } else { + foreach ($r in $ur) { $rr = "$r"; $out += $(if ($script:calcRestrictMap[$rr]) { $script:calcRestrictMap[$rr] } else { $rr }) } + } + return ,$out +} + +function Emit-RestrictBlock { + param([string]$tag, $ur, [string]$indent) + $r = Get-RestrictList $ur + if ($r.Count -eq 0) { return } + X "$indent" + foreach ($k in @('field','condition','group','order')) { if ($r -contains $k) { X "$indent`ttrue" } } + X "$indent" +} + # --- 5. Type emitter --- $script:formTypeSynonyms = New-Object System.Collections.Hashtable @@ -5626,6 +5651,9 @@ function Emit-Attributes { Emit-MLItems -val $fld.title -indent "$si`t`t" X "$si`t" } + # Ограничения использования поля — после title, перед presentationExpression (порядок исходника) + Emit-RestrictBlock 'useRestriction' $fld.useRestriction "$si`t" + Emit-RestrictBlock 'attributeUseRestriction' $fld.attributeUseRestriction "$si`t" # presentationExpression поля — перед valueType (порядок исходника) if ($fld.presentationExpression) { X "$si`t$(Esc-Xml "$($fld.presentationExpression)")" } # valueType поля набора (тип значения; вычисляемые/кастомные поля) @@ -5636,6 +5664,8 @@ function Emit-Attributes { foreach ($prop in $fld.appearance.PSObject.Properties) { Emit-AppearanceValue -key $prop.Name -val $prop.Value -indent "$si`t`t" } X "$si`t" } + # inputParameters поля (связь по параметрам выбора) — в конце + if ($fld.inputParameters) { Emit-DLInputParameters -ip $fld.inputParameters -indent "$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 2a5ef656..ecf281fc 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.141 — Compile 1C managed form from JSON or object metadata +# form-compile v1.142 — Compile 1C managed form from JSON or object metadata # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills import argparse import copy @@ -1832,6 +1832,31 @@ def emit_calc_fields(lines, calc_fields, indent): lines.append(f'{indent}') +# Ограничения использования поля/вычисляемого поля (useRestriction / attributeUseRestriction). +# Значение: объект {field?,condition?,group?,order?} | флаг-строка "#noField …" | массив. +def parse_restrict(ur): + if not ur: + return [] + if isinstance(ur, dict): + return [k for k in ('field', 'condition', 'group', 'order') if ur.get(k)] + if isinstance(ur, str): + return [_CALC_RESTRICT_MAP.get(t.strip().lstrip('#'), t.strip().lstrip('#')) for t in ur.split() if t.strip()] + if isinstance(ur, list): + return [_CALC_RESTRICT_MAP.get(str(r), str(r)) for r in ur] + return [] + + +def emit_restrict_block(lines, tag, ur, indent): + r = parse_restrict(ur) + if not r: + return + lines.append(f'{indent}') + for k in ('field', 'condition', 'group', 'order'): + if k in r: + lines.append(f'{indent}\ttrue') + lines.append(f'{indent}') + + def emit_conditional_appearance(lines, items, indent, block_view_mode=None, block_user_setting_id=None, wrap_tag='dcsset:conditionalAppearance'): has_items = bool(items) and len(items) > 0 has_block_meta = (block_view_mode is not None) or (block_user_setting_id is not None) @@ -5373,6 +5398,9 @@ 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') + # Ограничения использования поля — после title, перед presentationExpression + emit_restrict_block(lines, 'useRestriction', fld.get('useRestriction'), f'{si}\t') + emit_restrict_block(lines, 'attributeUseRestriction', fld.get('attributeUseRestriction'), f'{si}\t') # presentationExpression поля — перед valueType (порядок исходника) if fld.get('presentationExpression'): lines.append(f'{si}\t{esc_xml(str(fld["presentationExpression"]))}') @@ -5385,6 +5413,9 @@ def emit_attributes(lines, attrs, indent, conditional_appearance=None): for ak, av in fld['appearance'].items(): emit_appearance_value(lines, ak, av, f'{si}\t\t') lines.append(f'{si}\t') + # inputParameters поля (связь по параметрам выбора) — в конце + if fld.get('inputParameters'): + emit_dl_input_parameters(lines, fld['inputParameters'], 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 2ae332ec..28a96037 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.114 — Decompile 1C managed Form.xml to JSON DSL (draft) +# form-decompile v0.115 — Decompile 1C managed Form.xml to JSON DSL (draft) # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills # ВНИМАНИЕ: раундтрип не гарантируется. Навык исключён из авто-использования моделью. param( @@ -1246,6 +1246,14 @@ function Decompile-Type { return ($parts -join ' | ') } +# Ограничения использования (useRestriction/attributeUseRestriction) → объект {field?,condition?,group?,order?}. +function Build-RestrictObj { + param($node) + $r = [ordered]@{} + foreach ($k in 'field','condition','group','order') { if ((Get-Child $node $k) -eq 'true') { $r[$k] = $true } } + return $r +} + # Вычисляемое поле DataSet динамического списка () → объектная модель. # Зеркало эмиссии form-compile (Emit-CalcFields). Грамматика — как skd calculatedFields, # + форм-extras presentationExpression/orderExpression (в namespace dcscommon). @@ -2619,6 +2627,12 @@ if ($attrsNode) { 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 } } + $furNode = $fn.SelectSingleNode("dcssch:useRestriction", $ns) + if ($furNode) { $fur = Build-RestrictObj $furNode; if ($fur.Count -gt 0) { $fo['useRestriction'] = $fur } } + $faurNode = $fn.SelectSingleNode("dcssch:attributeUseRestriction", $ns) + if ($faurNode) { $faur = Build-RestrictObj $faurNode; if ($faur.Count -gt 0) { $fo['attributeUseRestriction'] = $faur } } + $fipNode = $fn.SelectSingleNode("dcssch:inputParameters", $ns) + if ($fipNode) { $fip = Build-DLInputParameters $fipNode; if (@($fip).Count -gt 0) { $fo['inputParameters'] = $fip } } [void]$fields.Add($fo) } $so['fields'] = @($fields) diff --git a/docs/form-dsl-spec.md b/docs/form-dsl-spec.md index 85194cbd..ce48b68b 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?, presentationExpression?, appearance?, nested? }` — для переопределения свойств поля. `valueType` — тип значения (грамматика типа). `presentationExpression` — выражение представления поля (строка). `appearance` — оформление/формат поля: объект `{ Параметр: Значение }` (та же грамматика, что `appearance` условного оформления, напр. `{ "Формат": "ДЛ=9", "ЦветТекста": "web:Gray" }`). `nested: true` помечает поле-вложенный набор (`DataSetFieldNestedDataSet` = реквизит табличной части объекта; дефолт — `DataSetFieldField`). Обычно поля выводятся из запроса автоматически | +| `fields` | array | Явные поля набора (редко): `{ field, dataPath?, title?, useRestriction?, attributeUseRestriction?, valueType?, presentationExpression?, appearance?, inputParameters?, nested? }` — для переопределения свойств поля. `useRestriction`/`attributeUseRestriction` — ограничения использования: объект `{ field?, condition?, group?, order? }` (bool) или флаг-строка `"#noField #noFilter #noGroup #noOrder"`. `valueType` — тип значения (грамматика типа). `presentationExpression` — выражение представления поля (строка). `appearance` — оформление/формат поля: объект `{ Параметр: Значение }` (та же грамматика, что `appearance` условного оформления, напр. `{ "Формат": "ДЛ=9", "ЦветТекста": "web:Gray" }`). `inputParameters` — связь по параметрам выбора (как у параметра дин-списка). `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 9a645595..cfa135d4 100644 --- a/tests/skills/cases/form-compile/dynamic-list-form.json +++ b/tests/skills/cases/form-compile/dynamic-list-form.json @@ -22,6 +22,8 @@ "parameters": [ { "name": "ВыборТовара", "type": "CatalogRef.Товары", "valueListAllowed": true, "value": null } ], "fields": [ { "field": "Code", "title": "Код", "presentationExpression": "\"№\" + Code", + "useRestriction": { "condition": true, "group": true }, + "attributeUseRestriction": { "field": true }, "appearance": { "Формат": "ДЛ=9", "ЦветТекста": "web:Gray" } } ], "grouping": "Description > Code", 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 4cbcd771..b9d743b6 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 @@ -109,6 +109,13 @@ Код + + true + true + + + true + "№" + Code