feat(form-decompile,form-compile): StandardBeginningDate в значении фильтра — структурно {variant, date?}

Значение фильтра типа v8:StandardBeginningDate (стандартная дата начала) серилизуется
структурно: <v8:variant xsi:type="v8:StandardBeginningDateVariant">Custom</v8:variant>
+ <v8:date>… (Custom несёт дату; именованные варианты — без). Компилятор эмитил
плоскую склейку InnerText (Custom3999-12-31T23:59:59), декомпилятор брал
сцепленный текст. Корпус 8.3.24: 307 случаев (Custom 280 с датой, BeginningOfThisDay
23, …Week 3, …Year 1; StandardEndDate/StandardPeriod как значение фильтра не
встречаются, но обработаны симметрично).

Не «забытый порт» — skd-decompile тоже не структурирует SBD в filter right (только в
dataParameters). DSL: value = {variant, date?} + valueType="v8:StandardBeginningDate".
Декомпилятор Get-FilterValueWithType читает variant/date; компилятор Emit-FilterItem
эмитит структурно (variant xsi:type выводится из valueType). Зеркало py.

Форма УправлениеОбменом (ДатаЗакрытия = SBD, op Equal): SBD-потерь 0 (остаток diff —
несвязанный TitleFont/FooterText). Кейс input-fields (+CA фильтр SBD Custom+date и
именованный вариант) сертифицирован в 1С, round-trip декомпиляции подтверждён.
Регресс 43/43.

ОТДЕЛЬНАЯ НАХОДКА (не в этом коммите): операторы Filled/NotFilled несут
тип-зависимый плейсхолдер <dcsset:right> (пустой xs:string для строк, SBD с дефолт.
датой для дат), который декомпилятор дропает как беззначный — нужен отдельный фикс.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
Nick Shirokov
2026-06-11 18:25:24 +03:00
parent 459d3feb69
commit 3ce71d436f
6 changed files with 92 additions and 5 deletions
@@ -1,4 +1,4 @@
# form-compile v1.116 — Compile 1C managed form from JSON or object metadata
# form-compile v1.117 — Compile 1C managed form from JSON or object metadata
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
param(
[string]$JsonPath,
@@ -1718,6 +1718,20 @@ function Emit-FilterItem {
X "$indent`t<dcsset:right xsi:type=`"$vt`">$vStr</dcsset:right>"
}
}
} elseif ($null -ne $item.value -and "$($item.valueType)" -match 'Standard(Beginning|End)Date$' -and (($item.value -is [PSCustomObject]) -or ($item.value -is [System.Collections.IDictionary]))) {
# Стандартная дата начала/окончания: структурное значение {variant, date?}.
# Custom несёт <v8:date>; именованные варианты (BeginningOfThisDay/…) — без даты.
$sdType = "$($item.valueType)" -replace '^v8:',''
$sv = $item.value
$variant = if ($sv -is [PSCustomObject]) { "$($sv.variant)" } else { "$($sv['variant'])" }
$hasDate = if ($sv -is [PSCustomObject]) { [bool]$sv.PSObject.Properties['date'] } else { $sv.Contains('date') }
X "$indent`t<dcsset:right xsi:type=`"v8:$sdType`">"
X "$indent`t`t<v8:variant xsi:type=`"v8:${sdType}Variant`">$(Esc-Xml $variant)</v8:variant>"
if ($hasDate) {
$dateV = if ($sv -is [PSCustomObject]) { "$($sv.date)" } else { "$($sv['date'])" }
X "$indent`t`t<v8:date>$(Esc-Xml $dateV)</v8:date>"
}
X "$indent`t</dcsset:right>"
} elseif ($null -ne $item.value) {
$vt = if ($item.valueType) { "$($item.valueType)" } else { "" }
if (-not $vt) {
@@ -1,5 +1,5 @@
#!/usr/bin/env python3
# form-compile v1.116 — Compile 1C managed form from JSON or object metadata
# form-compile v1.117 — Compile 1C managed form from JSON or object metadata
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
import argparse
import copy
@@ -1465,6 +1465,15 @@ def emit_filter_item(lines, item, indent):
vt = _value_type_for(v, item.get('valueType'))
v_str = str(v).lower() if isinstance(v, bool) else esc_xml(str(v))
lines.append(f'{indent}\t<dcsset:right xsi:type="{vt}">{v_str}</dcsset:right>')
elif val is not None and isinstance(val, dict) and re.search(r'Standard(Beginning|End)Date$', str(item.get('valueType') or '')):
# Стандартная дата начала/окончания: структурное значение {variant, date?}.
# Custom несёт <v8:date>; именованные варианты (BeginningOfThisDay/…) — без даты.
sd_type = re.sub(r'^v8:', '', str(item['valueType']))
lines.append(f'{indent}\t<dcsset:right xsi:type="v8:{sd_type}">')
lines.append(f'{indent}\t\t<v8:variant xsi:type="v8:{sd_type}Variant">{esc_xml(str(val.get("variant", "")))}</v8:variant>')
if 'date' in val:
lines.append(f'{indent}\t\t<v8:date>{esc_xml(str(val["date"]))}</v8:date>')
lines.append(f'{indent}\t</dcsset:right>')
elif val is not None:
vt = _value_type_for(val, item.get('valueType'))
v_str = str(val).lower() if isinstance(val, bool) else esc_xml(str(val))
@@ -1,4 +1,4 @@
# form-decompile v0.92 — Decompile 1C managed Form.xml to JSON DSL (draft)
# form-decompile v0.93 — Decompile 1C managed Form.xml to JSON DSL (draft)
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
# ВНИМАНИЕ: раундтрип не гарантируется. Навык исключён из авто-использования моделью.
param(
@@ -465,6 +465,15 @@ function Get-FilterValueWithType {
if ($vType -eq 'LocalStringType') {
return @{ value = (Get-MLText $valNode); type = $rawType }
}
# Стандартная дата начала/окончания — структурное значение {variant, date?}
# (Custom несёт <v8:date>; именованные варианты — без даты). Иначе InnerText склеивал variant+date.
if ($vType -eq 'StandardBeginningDate' -or $vType -eq 'StandardEndDate') {
$variantN = $valNode.SelectSingleNode("v8:variant", $ns)
$dateN = $valNode.SelectSingleNode("v8:date", $ns)
$o = [ordered]@{ variant = if ($variantN) { $variantN.InnerText } else { '' } }
if ($dateN) { $o['date'] = $dateN.InnerText }
return @{ value = $o; type = $rawType }
}
$txt = $valNode.InnerText
if (-not $txt) { return @{ value = '_'; type = $rawType } }
if ($vType -eq 'boolean') { return @{ value = ($txt -eq 'true'); type = $rawType } }
+1
View File
@@ -975,6 +975,7 @@ Forgiving-синонимы типа: XML-имя (`SpreadSheetDocumentField`) и
- **order** — строка `"Поле"` (asc) / `"Поле desc"` (синонимы `убыв`/`desc`, `возр`/`asc`) / `"Auto"`, либо объект `{ field, direction?, use?, viewMode? }`.
- **filter** — shorthand `"Поле оператор значение @флаги"` (`@off`, `@user`, `@quickAccess`, `@normal`, `@inaccessible`; `_` = пусто) или объект `{ field, op, value?, use?, userSettingID? }` или группа `{ group: "And"|"Or"|"Not", items: [...] }`.
- **Стандартная дата** (`valueType: "v8:StandardBeginningDate"`): значение — объект `{ variant, date? }`. `variant: "Custom"` несёт `date` (ISO-дата); именованные варианты (`BeginningOfThisDay`/`BeginningOfThisWeek`/`BeginningOfThisYear`/…) — без `date`. Эмитится структурно (`<v8:variant>`+`<v8:date>`), не плоской строкой.
- **conditionalAppearance** — объект `{ selection?, filter?, appearance?, presentation?, viewMode?, userSettingID?, use? }`. `appearance` — словарь «параметр: значение» платформы (`ЦветТекста`, `ЦветФона`, `Шрифт` и т.п.).
- Значение текстовых параметров (`Текст`/`Заголовок`/`Формат`) ведётся **по форме значения**: голая строка → плоский `xs:string` (нелокализованный литерал; `""` → самозакрывающийся тег); объект `{ru,en}` → локализуемый `LocalStringType`; объект `{field:"путь"}` → ссылка на поле компоновки (`dcscor:Field`). (В отличие от `title`/`tooltip`, где голая строка = `LocalStringType` — здесь это намеренное scoped-различие: платформа хранит обе формы, и их надо различать.)
@@ -66,14 +66,17 @@
{ "name": "ФлагПлатформенный", "type": "boolean" },
{ "name": "ФлагЯвный", "type": "boolean" },
{ "name": "ФлагТумблер", "type": "boolean" },
{ "name": "ЧисловоеПоле", "type": "number(10,2)" }
{ "name": "ЧисловоеПоле", "type": "number(10,2)" },
{ "name": "ДатаПоля", "type": "dateTime" }
],
"conditionalAppearance": [
{ "selection": ["ОбычноеПоле"], "filter": ["ЧисловоеПоле > 100"], "appearance": { "ЦветФона": "style:FormBackColor", "Текст": "Плоский текст" },
"presentation": { "ru": "Подсветка", "en": "Highlight" } },
{ "filter": ["ЧисловоеПоле = 0"], "appearance": { "Текст": { "ru": "Локализованный", "en": "Localized" } } },
{ "filter": ["ЧисловоеПоле < 0"], "appearance": { "Текст": "" } },
{ "filter": ["ЧисловоеПоле = 1"], "appearance": { "Текст": { "field": "ОбычноеПоле" } } }
{ "filter": ["ЧисловоеПоле = 1"], "appearance": { "Текст": { "field": "ОбычноеПоле" } } },
{ "filter": [{ "field": "ДатаПоля", "op": "=", "value": { "variant": "Custom", "date": "3999-12-31T23:59:59" }, "valueType": "v8:StandardBeginningDate" }], "appearance": { "ЦветТекста": "web:Gray" } },
{ "filter": [{ "field": "ДатаПоля", "op": ">=", "value": { "variant": "BeginningOfThisDay" }, "valueType": "v8:StandardBeginningDate" }], "appearance": { "ЦветТекста": "web:Silver" } }
]
}
}
@@ -610,6 +610,20 @@
</v8:NumberQualifiers>
</Type>
</Attribute>
<Attribute name="ДатаПоля" id="57">
<Title>
<v8:item>
<v8:lang>ru</v8:lang>
<v8:content>Дата поля</v8:content>
</v8:item>
</Title>
<Type>
<v8:Type>xs:dateTime</v8:Type>
<v8:DateQualifiers>
<v8:DateFractions>DateTime</v8:DateFractions>
</v8:DateQualifiers>
</Type>
</Attribute>
<ConditionalAppearance>
<dcsset:item>
<dcsset:selection>
@@ -702,6 +716,43 @@
</dcscor:item>
</dcsset:appearance>
</dcsset:item>
<dcsset:item>
<dcsset:selection/>
<dcsset:filter>
<dcsset:item xsi:type="dcsset:FilterItemComparison">
<dcsset:left xsi:type="dcscor:Field">ДатаПоля</dcsset:left>
<dcsset:comparisonType>Equal</dcsset:comparisonType>
<dcsset:right xsi:type="v8:StandardBeginningDate">
<v8:variant xsi:type="v8:StandardBeginningDateVariant">Custom</v8:variant>
<v8:date>3999-12-31T23:59:59</v8:date>
</dcsset:right>
</dcsset:item>
</dcsset:filter>
<dcsset:appearance>
<dcscor:item xsi:type="dcsset:SettingsParameterValue">
<dcscor:parameter>ЦветТекста</dcscor:parameter>
<dcscor:value xsi:type="v8ui:Color">web:Gray</dcscor:value>
</dcscor:item>
</dcsset:appearance>
</dcsset:item>
<dcsset:item>
<dcsset:selection/>
<dcsset:filter>
<dcsset:item xsi:type="dcsset:FilterItemComparison">
<dcsset:left xsi:type="dcscor:Field">ДатаПоля</dcsset:left>
<dcsset:comparisonType>GreaterOrEqual</dcsset:comparisonType>
<dcsset:right xsi:type="v8:StandardBeginningDate">
<v8:variant xsi:type="v8:StandardBeginningDateVariant">BeginningOfThisDay</v8:variant>
</dcsset:right>
</dcsset:item>
</dcsset:filter>
<dcsset:appearance>
<dcscor:item xsi:type="dcsset:SettingsParameterValue">
<dcscor:parameter>ЦветТекста</dcscor:parameter>
<dcscor:value xsi:type="v8ui:Color">web:Silver</dcscor:value>
</dcscor:item>
</dcsset:appearance>
</dcsset:item>
</ConditionalAppearance>
</Attributes>
</Form>