feat(form-decompile,form-compile): ConditionalAppearance формы из ring-3 (переиспользование DCS-грамматики)

Снят fast-fail на ConditionalAppearance (1304 формы, 4%). Структура — та же
DCS-грамматика, что settings.conditionalAppearance дин-списка, поэтому
переиспользованы Build-ConditionalAppearance (декомпилятор) и
Emit-ConditionalAppearance (компилятор) как есть.

Отличия от настроек списка: тег-обёртка <ConditionalAppearance> (без dcsset:,
параметр wrapTag) + нет блок-мета viewMode/userSettingID + размещение (последний
child <Attributes>, не отдельный Form-child). Форменный ключ conditionalAppearance
(selection/filter/appearance/presentation). Scope в формах не встречается
(0/6186) → fail-ring3 только при scope.

Заодно фикс: мультиязык-presentation элемента CA → xsi:type="v8:LocalStringType"
(был голый <dcsset:presentation>; чинит и settings CA path).

Выборка 2.17: ring3 7→4 (остаток — только chart-семейство), match 211→214,
CA-формы бит-в-бит. Зеркало py байт-в-байт, кейс input-fields
(+conditionalAppearance: selection+filter+appearance+presentation)
сертифицирован загрузкой в 1С. Регресс 40/40 (ps1+py).

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
Nick Shirokov
2026-06-09 22:08:14 +03:00
parent ca96a7413a
commit 5ebc02d0b3
6 changed files with 117 additions and 18 deletions
@@ -1,4 +1,4 @@
# form-compile v1.100 — Compile 1C managed form from JSON or object metadata
# form-compile v1.101 — Compile 1C managed form from JSON or object metadata
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
param(
[string]$JsonPath,
@@ -1890,11 +1890,11 @@ function Emit-AppearanceValue {
}
function Emit-ConditionalAppearance {
param($items, [string]$indent, $blockViewMode = $null, $blockUserSettingID = $null)
param($items, [string]$indent, $blockViewMode = $null, $blockUserSettingID = $null, [string]$wrapTag = 'dcsset:conditionalAppearance')
$hasItems = $items -and $items.Count -gt 0
$hasBlockMeta = ($null -ne $blockViewMode) -or ($null -ne $blockUserSettingID)
if (-not $hasItems -and -not $hasBlockMeta) { return }
X "$indent<dcsset:conditionalAppearance>"
X "$indent<$wrapTag>"
foreach ($ca in $items) {
X "$indent`t<dcsset:item>"
if ($ca.use -eq $false) { X "$indent`t`t<dcsset:use>false</dcsset:use>" }
@@ -1915,7 +1915,12 @@ function Emit-ConditionalAppearance {
X "$indent`t`t</dcsset:appearance>"
}
if ($ca.presentation) {
if ($ca.presentation -is [hashtable] -or $ca.presentation -is [System.Collections.IDictionary] -or $ca.presentation -is [PSCustomObject]) { Emit-MLText -tag "dcsset:presentation" -text $ca.presentation -indent "$indent`t`t" }
if ($ca.presentation -is [hashtable] -or $ca.presentation -is [System.Collections.IDictionary] -or $ca.presentation -is [PSCustomObject]) {
# Мультиязык → LocalStringType (платформа объявляет тип у локализованного presentation)
X "$indent`t`t<dcsset:presentation xsi:type=`"v8:LocalStringType`">"
Emit-MLItems -val $ca.presentation -indent "$indent`t`t`t"
X "$indent`t`t</dcsset:presentation>"
}
else { X "$indent`t`t<dcsset:presentation xsi:type=`"xs:string`">$(Esc-Xml "$($ca.presentation)")</dcsset:presentation>" }
}
if ($ca.viewMode) { X "$indent`t`t<dcsset:viewMode>$(Esc-Xml "$($ca.viewMode)")</dcsset:viewMode>" }
@@ -1942,7 +1947,7 @@ function Emit-ConditionalAppearance {
$uid = if ("$blockUserSettingID" -eq 'auto') { New-Guid-String } else { "$blockUserSettingID" }
X "$indent`t<dcsset:userSettingID>$(Esc-Xml $uid)</dcsset:userSettingID>"
}
X "$indent</dcsset:conditionalAppearance>"
X "$indent</$wrapTag>"
}
# --- 5. Type emitter ---
@@ -4674,10 +4679,18 @@ function Emit-DLParameters {
}
function Emit-Attributes {
param($attrs, [string]$indent)
param($attrs, [string]$indent, $conditionalAppearance = $null)
$hasCA = $conditionalAppearance -and @($conditionalAppearance).Count -gt 0
# Платформа ВСЕГДА эмитит <Attributes> (100% корпуса; 162 формы — пустой <Attributes/>).
if (-not $attrs -or $attrs.Count -eq 0) { X "$indent<Attributes/>"; return }
if ((-not $attrs -or $attrs.Count -eq 0) -and -not $hasCA) { X "$indent<Attributes/>"; return }
if (-not $attrs -or $attrs.Count -eq 0) {
# Нет реквизитов, но есть условное оформление (последний child <Attributes>)
X "$indent<Attributes>"
Emit-ConditionalAppearance -items $conditionalAppearance -indent "$indent`t" -wrapTag 'ConditionalAppearance'
X "$indent</Attributes>"
return
}
X "$indent<Attributes>"
$seenAttrs = @{}
@@ -4890,6 +4903,8 @@ function Emit-Attributes {
X "$indent`t</Attribute>"
}
# Условное оформление формы — последний child <Attributes> (та же DCS-грамматика, что settings CA)
Emit-ConditionalAppearance -items $conditionalAppearance -indent "$indent`t" -wrapTag 'ConditionalAppearance'
X "$indent</Attributes>"
}
@@ -5465,7 +5480,7 @@ if ($def.elements -and $def.elements.Count -gt 0) {
}
# 12g. Attributes
Emit-Attributes -attrs $def.attributes -indent "`t"
Emit-Attributes -attrs $def.attributes -indent "`t" -conditionalAppearance $def.conditionalAppearance
# 12h. Parameters
Emit-Parameters -params $def.parameters -indent "`t"
@@ -1,5 +1,5 @@
#!/usr/bin/env python3
# form-compile v1.100 — Compile 1C managed form from JSON or object metadata
# form-compile v1.101 — Compile 1C managed form from JSON or object metadata
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
import argparse
import copy
@@ -1640,12 +1640,12 @@ def emit_appearance_value(lines, key, val, indent):
lines.append(f'{indent}</dcscor:item>')
def emit_conditional_appearance(lines, items, indent, block_view_mode=None, block_user_setting_id=None):
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)
if not has_items and not has_block_meta:
return
lines.append(f'{indent}<dcsset:conditionalAppearance>')
lines.append(f'{indent}<{wrap_tag}>')
for ca in (items or []):
lines.append(f'{indent}\t<dcsset:item>')
if ca.get('use') is False:
@@ -1670,7 +1670,10 @@ def emit_conditional_appearance(lines, items, indent, block_view_mode=None, bloc
lines.append(f'{indent}\t\t</dcsset:appearance>')
if ca.get('presentation'):
if isinstance(ca['presentation'], dict):
emit_mltext(lines, f'{indent}\t\t', 'dcsset:presentation', ca['presentation'])
# Мультиязык → LocalStringType (платформа объявляет тип у локализованного presentation)
lines.append(f'{indent}\t\t<dcsset:presentation xsi:type="v8:LocalStringType">')
emit_ml_items(lines, f'{indent}\t\t\t', ca['presentation'])
lines.append(f'{indent}\t\t</dcsset:presentation>')
else:
lines.append(f'{indent}\t\t<dcsset:presentation xsi:type="xs:string">{esc_xml(str(ca["presentation"]))}</dcsset:presentation>')
if ca.get('viewMode'):
@@ -1695,7 +1698,7 @@ def emit_conditional_appearance(lines, items, indent, block_view_mode=None, bloc
if block_user_setting_id is not None:
uid = new_uuid() if str(block_user_setting_id) == 'auto' else str(block_user_setting_id)
lines.append(f'{indent}\t<dcsset:userSettingID>{esc_xml(uid)}</dcsset:userSettingID>')
lines.append(f'{indent}</dcsset:conditionalAppearance>')
lines.append(f'{indent}</{wrap_tag}>')
def write_utf8_bom(path, content):
@@ -4388,11 +4391,18 @@ def emit_dl_parameters(lines, params, indent):
emit_dl_parameter(lines, p, parsed, indent)
def emit_attributes(lines, attrs, indent):
def emit_attributes(lines, attrs, indent, conditional_appearance=None):
has_ca = bool(conditional_appearance) and len(conditional_appearance) > 0
# Платформа ВСЕГДА эмитит <Attributes> (100% корпуса; 162 формы — пустой <Attributes/>).
if not attrs or len(attrs) == 0:
if (not attrs or len(attrs) == 0) and not has_ca:
lines.append(f'{indent}<Attributes/>')
return
if not attrs or len(attrs) == 0:
# Нет реквизитов, но есть условное оформление (последний child <Attributes>)
lines.append(f'{indent}<Attributes>')
emit_conditional_appearance(lines, conditional_appearance, f'{indent}\t', wrap_tag='ConditionalAppearance')
lines.append(f'{indent}</Attributes>')
return
lines.append(f'{indent}<Attributes>')
seen_attrs = set()
@@ -4595,6 +4605,8 @@ def emit_attributes(lines, attrs, indent):
lines.append(f'{inner}</Settings>')
lines.append(f'{indent}\t</Attribute>')
# Условное оформление формы — последний child <Attributes> (та же DCS-грамматика, что settings CA)
emit_conditional_appearance(lines, conditional_appearance, f'{indent}\t', wrap_tag='ConditionalAppearance')
lines.append(f'{indent}</Attributes>')
@@ -5297,7 +5309,7 @@ def main():
lines.append('\t</ChildItems>')
# Attributes
emit_attributes(lines, defn.get('attributes'), '\t')
emit_attributes(lines, defn.get('attributes'), '\t', conditional_appearance=defn.get('conditionalAppearance'))
# Parameters
emit_parameters(lines, defn.get('parameters'), '\t')
@@ -1,4 +1,4 @@
# form-decompile v0.76 — Decompile 1C managed Form.xml to JSON DSL (draft)
# form-decompile v0.77 — Decompile 1C managed Form.xml to JSON DSL (draft)
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
# ВНИМАНИЕ: раундтрип не гарантируется. Навык исключён из авто-использования моделью.
param(
@@ -145,7 +145,8 @@ function Fail-Ring3 {
[Console]::Error.WriteLine("Для точечной работы с этой формой используй /form-edit.")
exit 3
}
foreach ($el in $xmlDoc.SelectNodes("//*[local-name()='ConditionalAppearance']")) { Fail-Ring3 -kind "ConditionalAppearance" -loc "form/ConditionalAppearance" }
# ConditionalAppearance со scope (привязка к области) пока не воспроизводим — fail-ring3 только в этом случае.
foreach ($el in $xmlDoc.SelectNodes("//*[local-name()='ConditionalAppearance']/*[local-name()='item']/*[local-name()='scope'][node()]")) { Fail-Ring3 -kind "ConditionalAppearance со scope" -loc "form/ConditionalAppearance/item/scope" }
# --- 1c. Compact JSON serializer (созвучно skd-decompile: 2-проб. indent, inline в пределах lineLimit) ---
function Convert-StringToJsonLiteral {
@@ -2099,6 +2100,16 @@ if ($attrsNode) {
if ($attrs.Count -gt 0) { $dsl['attributes'] = @($attrs) }
}
# conditionalAppearance формы (<ConditionalAppearance> — последний child <Attributes>;
# та же DCS-грамматика, что settings.conditionalAppearance → переиспользуем Build-ConditionalAppearance)
if ($attrsNode) {
$caNode = $attrsNode.SelectSingleNode("lf:ConditionalAppearance", $ns)
if ($caNode) {
$ca = Build-ConditionalAppearance -caNode $caNode -loc "form/conditionalAppearance"
if (@($ca).Count -gt 0) { $dsl['conditionalAppearance'] = @($ca) }
}
}
# parameters
$parsNode = $root.SelectSingleNode("lf:Parameters", $ns)
if ($parsNode) {
+25
View File
@@ -985,6 +985,31 @@ Forgiving-синонимы типа: XML-имя (`SpreadSheetDocumentField`) и
---
## 7c. conditionalAppearance — условное оформление формы
Форменный ключ `conditionalAppearance` (XML `<ConditionalAppearance>` — последний child `<Attributes>`).
**Грамматика идентична `settings.conditionalAppearance` дин-списка** (DCS — см. §«order/filter/conditionalAppearance»):
массив объектов `{ selection?, filter?, appearance?, presentation?, viewMode?, userSettingID?, use? }`.
```json
"conditionalAppearance": [
{ "selection": ["ОбычноеПоле"], "filter": ["ЧисловоеПоле > 100"],
"appearance": { "ЦветФона": "style:FormBackColor" },
"presentation": { "ru": "Подсветка", "en": "Highlight" } }
]
```
- `selection` — массив имён форматируемых полей (`<dcsset:field>`).
- `filter` — условие (filter-shorthand, как в СКД).
- `appearance` — словарь «параметр-DCS: значение» (рус. verbatim: `ЦветТекста`/`ЦветФона`/`Шрифт`/…). Цвет → `v8ui:Color`.
- `presentation` — мультиязык → `xsi:type="v8:LocalStringType"`.
Декомпилятор/компилятор переиспользуют `Build-ConditionalAppearance`/`Emit-ConditionalAppearance` настроек списка
(отличие — тег-обёртка `ConditionalAppearance` без `dcsset:` и без блок-мета). `scope` (привязка к области) в формах
не встречается; форма со `scope` → fail-ring3.
---
## 8. Система типов (shorthand)
### Примитивные типы
@@ -67,6 +67,10 @@
{ "name": "ФлагЯвный", "type": "boolean" },
{ "name": "ФлагТумблер", "type": "boolean" },
{ "name": "ЧисловоеПоле", "type": "number(10,2)" }
],
"conditionalAppearance": [
{ "selection": ["ОбычноеПоле"], "filter": ["ЧисловоеПоле > 100"], "appearance": { "ЦветФона": "style:FormBackColor" },
"presentation": { "ru": "Подсветка", "en": "Highlight" } }
]
}
}
@@ -592,5 +592,37 @@
</v8:NumberQualifiers>
</Type>
</Attribute>
<ConditionalAppearance>
<dcsset:item>
<dcsset:selection>
<dcsset:item>
<dcsset:field>ОбычноеПоле</dcsset:field>
</dcsset:item>
</dcsset:selection>
<dcsset:filter>
<dcsset:item xsi:type="dcsset:FilterItemComparison">
<dcsset:left xsi:type="dcscor:Field">ЧисловоеПоле</dcsset:left>
<dcsset:comparisonType>Greater</dcsset:comparisonType>
<dcsset:right xsi:type="xs:decimal">100</dcsset:right>
</dcsset:item>
</dcsset:filter>
<dcsset:appearance>
<dcscor:item xsi:type="dcsset:SettingsParameterValue">
<dcscor:parameter>ЦветФона</dcscor:parameter>
<dcscor:value xsi:type="v8ui:Color">style:FormBackColor</dcscor:value>
</dcscor:item>
</dcsset:appearance>
<dcsset:presentation xsi:type="v8:LocalStringType">
<v8:item>
<v8:lang>ru</v8:lang>
<v8:content>Подсветка</v8:content>
</v8:item>
<v8:item>
<v8:lang>en</v8:lang>
<v8:content>Highlight</v8:content>
</v8:item>
</dcsset:presentation>
</dcsset:item>
</ConditionalAppearance>
</Attributes>
</Form>