From 3ce71d436f6c4d27e34ba9d771658538475f0ba4 Mon Sep 17 00:00:00 2001 From: Nick Shirokov Date: Thu, 11 Jun 2026 18:25:24 +0300 Subject: [PATCH] =?UTF-8?q?feat(form-decompile,form-compile):=20StandardBe?= =?UTF-8?q?ginningDate=20=D0=B2=20=D0=B7=D0=BD=D0=B0=D1=87=D0=B5=D0=BD?= =?UTF-8?q?=D0=B8=D0=B8=20=D1=84=D0=B8=D0=BB=D1=8C=D1=82=D1=80=D0=B0=20?= =?UTF-8?q?=E2=80=94=20=D1=81=D1=82=D1=80=D1=83=D0=BA=D1=82=D1=83=D1=80?= =?UTF-8?q?=D0=BD=D0=BE=20{variant,=20date=3F}?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Значение фильтра типа v8:StandardBeginningDate (стандартная дата начала) серилизуется структурно: Custom + … (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 несут тип-зависимый плейсхолдер (пустой xs:string для строк, SBD с дефолт. датой для дат), который декомпилятор дропает как беззначный — нужен отдельный фикс. Co-Authored-By: Claude Opus 4.8 (1M context) --- .../form-compile/scripts/form-compile.ps1 | 16 +++++- .../form-compile/scripts/form-compile.py | 11 +++- .../form-decompile/scripts/form-decompile.ps1 | 11 +++- docs/form-dsl-spec.md | 1 + .../cases/form-compile/input-fields.json | 7 ++- .../ПоляВвода/Forms/Форма/Ext/Form.xml | 51 +++++++++++++++++++ 6 files changed, 92 insertions(+), 5 deletions(-) diff --git a/.claude/skills/form-compile/scripts/form-compile.ps1 b/.claude/skills/form-compile/scripts/form-compile.ps1 index f24ec997..8ff9b6d9 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.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$vStr" } } + } 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 несёт ; именованные варианты (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" + X "$indent`t`t$(Esc-Xml $variant)" + if ($hasDate) { + $dateV = if ($sv -is [PSCustomObject]) { "$($sv.date)" } else { "$($sv['date'])" } + X "$indent`t`t$(Esc-Xml $dateV)" + } + X "$indent`t" } elseif ($null -ne $item.value) { $vt = if ($item.valueType) { "$($item.valueType)" } else { "" } if (-not $vt) { diff --git a/.claude/skills/form-compile/scripts/form-compile.py b/.claude/skills/form-compile/scripts/form-compile.py index 73abe5dd..b314520b 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.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{v_str}') + 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 несёт ; именованные варианты (BeginningOfThisDay/…) — без даты. + sd_type = re.sub(r'^v8:', '', str(item['valueType'])) + lines.append(f'{indent}\t') + lines.append(f'{indent}\t\t{esc_xml(str(val.get("variant", "")))}') + if 'date' in val: + lines.append(f'{indent}\t\t{esc_xml(str(val["date"]))}') + lines.append(f'{indent}\t') 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)) diff --git a/.claude/skills/form-decompile/scripts/form-decompile.ps1 b/.claude/skills/form-decompile/scripts/form-decompile.ps1 index 3bd38d04..7314a524 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.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 несёт ; именованные варианты — без даты). Иначе 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 } } diff --git a/docs/form-dsl-spec.md b/docs/form-dsl-spec.md index ced1e68b..101b9eed 100644 --- a/docs/form-dsl-spec.md +++ b/docs/form-dsl-spec.md @@ -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`. Эмитится структурно (``+``), не плоской строкой. - **conditionalAppearance** — объект `{ selection?, filter?, appearance?, presentation?, viewMode?, userSettingID?, use? }`. `appearance` — словарь «параметр: значение» платформы (`ЦветТекста`, `ЦветФона`, `Шрифт` и т.п.). - Значение текстовых параметров (`Текст`/`Заголовок`/`Формат`) ведётся **по форме значения**: голая строка → плоский `xs:string` (нелокализованный литерал; `""` → самозакрывающийся тег); объект `{ru,en}` → локализуемый `LocalStringType`; объект `{field:"путь"}` → ссылка на поле компоновки (`dcscor:Field`). (В отличие от `title`/`tooltip`, где голая строка = `LocalStringType` — здесь это намеренное scoped-различие: платформа хранит обе формы, и их надо различать.) diff --git a/tests/skills/cases/form-compile/input-fields.json b/tests/skills/cases/form-compile/input-fields.json index 85f4a877..3f1f21e3 100644 --- a/tests/skills/cases/form-compile/input-fields.json +++ b/tests/skills/cases/form-compile/input-fields.json @@ -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" } } ] } } diff --git a/tests/skills/cases/form-compile/snapshots/input-fields/DataProcessors/ПоляВвода/Forms/Форма/Ext/Form.xml b/tests/skills/cases/form-compile/snapshots/input-fields/DataProcessors/ПоляВвода/Forms/Форма/Ext/Form.xml index 6b9f9437..c958e9ca 100644 --- a/tests/skills/cases/form-compile/snapshots/input-fields/DataProcessors/ПоляВвода/Forms/Форма/Ext/Form.xml +++ b/tests/skills/cases/form-compile/snapshots/input-fields/DataProcessors/ПоляВвода/Forms/Форма/Ext/Form.xml @@ -610,6 +610,20 @@ + + + <v8:item> + <v8:lang>ru</v8:lang> + <v8:content>Дата поля</v8:content> + </v8:item> + + + xs:dateTime + + DateTime + + + @@ -702,6 +716,43 @@ + + + + + ДатаПоля + Equal + + Custom + 3999-12-31T23:59:59 + + + + + + ЦветТекста + web:Gray + + + + + + + + ДатаПоля + GreaterOrEqual + + BeginningOfThisDay + + + + + + ЦветТекста + web:Silver + + +