From c96bcc3566011c3df19374eb1275c6de6154030d Mon Sep 17 00:00:00 2001 From: Nick Shirokov Date: Thu, 11 Jun 2026 18:52:59 +0300 Subject: [PATCH] =?UTF-8?q?feat(form-decompile,form-compile):=20=D0=B4?= =?UTF-8?q?=D0=B0=D1=82=D0=B0=20=D0=B2=20=D1=84=D0=B8=D0=BB=D1=8C=D1=82?= =?UTF-8?q?=D1=80=D0=B5=20=3D=20StandardBeginningDate=20Custom=20(=D0=B3?= =?UTF-8?q?=D0=BE=D0=BB=D0=B0=D1=8F=20ISO-=D0=B4=D0=B0=D1=82=D0=B0=20?= =?UTF-8?q?=D0=BA=D0=B0=D0=BA=20=D1=88=D0=BE=D1=80=D1=82=D0=BA=D0=B0=D1=82?= =?UTF-8?q?)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Дата-значение фильтра платформой почти всегда хранится как StandardBeginningDate Custom, а не xs:dateTime (корпус 8.3.24: 268 SBD-Custom vs 2 xs:dateTime в dcsset:right). Добавлен естественный шорткат: голая ISO-дата без valueType → компилятор выводит SBD Custom+date. Работает и в shorthand-строке фильтра ("ДатаЗаказа > 2020-01-01T00:00:00"). Компилятор: Emit-FilterItem ловит дату без valueType → SBD Custom (раньше → xs:dateTime); Parse-FilterShorthand больше не ставит valueType=xs:dateTime для даты (оставляет вывод компилятору). Декомпилятор: SBD Custom+date → голая дата без valueType (компилятор восстановит), именованный вариант → строка+valueType, SED/нетипичное → объект+valueType. Escape для плоского xs:dateTime — явный valueType. Зеркало py. Кейс input-fields (+date-фильтр голой датой объектно и shorthand-строкой, +именованный вариант) сертифицирован в 1С, round-trip подтверждён (голые даты без valueType возвращаются). Регресс 43/43, SBD-корпус (40 форм) без регрессий (match 28). Co-Authored-By: Claude Opus 4.8 (1M context) --- .../form-compile/scripts/form-compile.ps1 | 19 +++++++++++++------ .../form-compile/scripts/form-compile.py | 17 +++++++++++------ .../form-decompile/scripts/form-decompile.ps1 | 18 ++++++++++++------ docs/form-dsl-spec.md | 7 ++++++- .../cases/form-compile/input-fields.json | 3 ++- .../ПоляВвода/Forms/Форма/Ext/Form.xml | 19 +++++++++++++++++++ 6 files changed, 63 insertions(+), 20 deletions(-) diff --git a/.claude/skills/form-compile/scripts/form-compile.ps1 b/.claude/skills/form-compile/scripts/form-compile.ps1 index 46c9fede..edc55841 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.118 — Compile 1C managed form from JSON or object metadata +# form-compile v1.119 — Compile 1C managed form from JSON or object metadata # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills param( [string]$JsonPath, @@ -1650,7 +1650,7 @@ function Parse-FilterShorthand { $valPart = if ($Matches[3]) { $Matches[3].Trim() } else { "" } if ($valPart -and $valPart -ne "_") { if ($valPart -eq "true" -or $valPart -eq "false") { $result.value = [bool]($valPart -eq "true"); $result["valueType"] = "xs:boolean" } - elseif ($valPart -match '^\d{4}-\d{2}-\d{2}T') { $result.value = $valPart; $result["valueType"] = "xs:dateTime" } + elseif ($valPart -match '^\d{4}-\d{2}-\d{2}T') { $result.value = $valPart } # дата без valueType → Emit-FilterItem выведет StandardBeginningDate Custom (дефолт даты в фильтре) elseif ($valPart -match '^\d+(\.\d+)?$') { $result.value = $valPart; $result["valueType"] = "xs:decimal" } elseif ($valPart -match '^(Перечисление|Справочник|ПланСчетов|Документ|ПланВидовХарактеристик|ПланВидовРасчета)\.') { $result.value = $valPart; $result["valueType"] = "dcscor:DesignTimeValue" } else { $result.value = $valPart; $result["valueType"] = "xs:string" } @@ -1718,16 +1718,23 @@ function Emit-FilterItem { X "$indent`t$vStr" } } - } elseif ($null -ne $item.value -and "$($item.valueType)" -match 'Standard(Beginning|End)Date$') { - # Стандартная дата начала/окончания. Две формы значения: + } elseif ($null -ne $item.value -and ( + "$($item.valueType)" -match 'Standard(Beginning|End)Date$' -or + (-not $item.valueType -and "$($item.value)" -match '^\d{4}-\d{2}-\d{2}T'))) { + # Стандартная дата начала/окончания. Формы значения: # объект {variant, date?} — полная (Custom несёт ); - # строка-вариант "BeginningOfThisDay" — short (именованный вариант без даты). - $sdType = "$($item.valueType)" -replace '^v8:','' + # строка-вариант "BeginningOfThisDay" — именованный вариант без даты; + # голая ISO-дата без valueType — шорткат для Custom+date (дата в фильтре платформой + # почти всегда хранится как StandardBeginningDate Custom, корпус 268 vs 2 xs:dateTime; + # явный valueType="xs:dateTime" → плоская дата, ветка ниже). + $sdType = if ($item.valueType) { "$($item.valueType)" -replace '^v8:','' } else { 'StandardBeginningDate' } $sv = $item.value if (($sv -is [PSCustomObject]) -or ($sv -is [System.Collections.IDictionary])) { $variant = if ($sv -is [PSCustomObject]) { "$($sv.variant)" } else { "$($sv['variant'])" } $hasDate = if ($sv -is [PSCustomObject]) { [bool]$sv.PSObject.Properties['date'] } else { $sv.Contains('date') } $dateV = if ($hasDate) { if ($sv -is [PSCustomObject]) { "$($sv.date)" } else { "$($sv['date'])" } } else { $null } + } elseif ("$sv" -match '^\d{4}-\d{2}-\d{2}T') { + $variant = 'Custom'; $hasDate = $true; $dateV = "$sv" } else { $variant = "$sv"; $hasDate = $false; $dateV = $null } diff --git a/.claude/skills/form-compile/scripts/form-compile.py b/.claude/skills/form-compile/scripts/form-compile.py index ad641bc4..2c2b7808 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.118 — Compile 1C managed form from JSON or object metadata +# form-compile v1.119 — Compile 1C managed form from JSON or object metadata # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills import argparse import copy @@ -1377,8 +1377,8 @@ def parse_filter_shorthand(s): result['value'] = (val_part == 'true') result['valueType'] = 'xs:boolean' elif re.match(r'^\d{4}-\d{2}-\d{2}T', val_part): + # дата без valueType → emit_filter_item выведет StandardBeginningDate Custom (дефолт даты в фильтре) result['value'] = val_part - result['valueType'] = 'xs:dateTime' elif re.match(r'^\d+(\.\d+)?$', val_part): result['value'] = val_part result['valueType'] = 'xs:decimal' @@ -1465,12 +1465,17 @@ 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 re.search(r'Standard(Beginning|End)Date$', str(item.get('valueType') or '')): - # Стандартная дата начала/окончания. Две формы: объект {variant, date?} (Custom несёт - # ) или строка-вариант "BeginningOfThisDay" (short, именованный без даты). - sd_type = re.sub(r'^v8:', '', str(item['valueType'])) + elif val is not None and ( + re.search(r'Standard(Beginning|End)Date$', str(item.get('valueType') or '')) or + (not item.get('valueType') and isinstance(val, str) and re.match(r'^\d{4}-\d{2}-\d{2}T', val))): + # Стандартная дата начала/окончания. Формы: объект {variant, date?} (Custom несёт ); + # строка-вариант "BeginningOfThisDay" (именованный без даты); голая ISO-дата без valueType — + # шорткат для Custom+date (дата в фильтре почти всегда SBD Custom, корпус 268 vs 2 xs:dateTime). + sd_type = re.sub(r'^v8:', '', str(item['valueType'])) if item.get('valueType') else 'StandardBeginningDate' if isinstance(val, dict): variant = str(val.get('variant', '')); date_v = val.get('date') + elif isinstance(val, str) and re.match(r'^\d{4}-\d{2}-\d{2}T', val): + variant = 'Custom'; date_v = val else: variant = str(val); date_v = None lines.append(f'{indent}\t') diff --git a/.claude/skills/form-decompile/scripts/form-decompile.ps1 b/.claude/skills/form-decompile/scripts/form-decompile.ps1 index e713f705..a986cf00 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.94 — Decompile 1C managed Form.xml to JSON DSL (draft) +# form-decompile v0.95 — Decompile 1C managed Form.xml to JSON DSL (draft) # Source: https://github.com/Nikolay-Shirokov/cc-1c-skills # ВНИМАНИЕ: раундтрип не гарантируется. Навык исключён из авто-использования моделью. param( @@ -465,15 +465,21 @@ function Get-FilterValueWithType { if ($vType -eq 'LocalStringType') { return @{ value = (Get-MLText $valNode); type = $rawType } } - # Стандартная дата начала/окончания. Custom несёт → объект {variant, date}; - # именованный вариант (BeginningOfThisDay/…) без даты → строка-вариант (short form). - # Иначе InnerText склеивал variant+date. + # Стандартная дата начала/окончания. Формы (от самой компактной): + # SBD Custom+date → голая ISO-дата без valueType (компилятор выводит SBD Custom — дефолт + # даты в фильтре, корпус 268 vs 2 xs:dateTime); именованный вариант → строка + valueType; + # SED Custom / нетипичное → объект {variant, date} + valueType. Иначе InnerText склеивал. if ($vType -eq 'StandardBeginningDate' -or $vType -eq 'StandardEndDate') { $variantN = $valNode.SelectSingleNode("v8:variant", $ns) $dateN = $valNode.SelectSingleNode("v8:date", $ns) $variantStr = if ($variantN) { $variantN.InnerText } else { '' } - if ($dateN) { return @{ value = [ordered]@{ variant = $variantStr; date = $dateN.InnerText }; type = $rawType } } - return @{ value = $variantStr; type = $rawType } + if ($dateN) { + if ($vType -eq 'StandardBeginningDate' -and $variantStr -eq 'Custom') { + return @{ value = $dateN.InnerText; type = $null } # голая дата = SBD Custom шорткат + } + return @{ value = [ordered]@{ variant = $variantStr; date = $dateN.InnerText }; type = $rawType } + } + return @{ value = $variantStr; type = $rawType } # именованный вариант } $txt = $valNode.InnerText if (-not $txt) { return @{ value = '_'; type = $rawType } } diff --git a/docs/form-dsl-spec.md b/docs/form-dsl-spec.md index 20d55539..68cb0a74 100644 --- a/docs/form-dsl-spec.md +++ b/docs/form-dsl-spec.md @@ -975,7 +975,12 @@ 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? }` ИЛИ короткая форма — строка-вариант. `Custom` несёт `date` (ISO) → объект `{ variant: "Custom", date: "…" }`; именованные варианты (`BeginningOfThisDay`/`BeginningOfThisWeek`/`BeginningOfThisYear`/…) — без даты → просто строка `"BeginningOfThisDay"`. Эмитится структурно (``+``). Декомпилятор: Custom → объект, именованный → строка. + - **Дата в фильтре = `StandardBeginningDate`** (так платформа хранит дату-значение почти всегда — корпус 268 vs 2 `xs:dateTime`). Формы значения (от компактной к полной): + - **голая ISO-дата** `"2020-01-01T00:00:00"` (без `valueType`) → `Custom` + эта дата. Работает и в shorthand: `"ДатаЗаказа > 2020-01-01T00:00:00"`. Это дефолт даты в фильтре. + - **строка-вариант** `"BeginningOfThisDay"` + `valueType: "v8:StandardBeginningDate"` — именованный вариант без даты (`BeginningOfThisWeek`/`BeginningOfThisYear`/…; имя ≠ дата, нужен `valueType`). + - **объект** `{ variant, date? }` + `valueType` — полная форма. + - **escape** для плоской `xs:dateTime`: явный `valueType: "xs:dateTime"`. + Эмитится структурно (``+``). Декомпилятор: Custom+date → голая дата; именованный → строка+valueType. - **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 c9ea1a91..9a1cce44 100644 --- a/tests/skills/cases/form-compile/input-fields.json +++ b/tests/skills/cases/form-compile/input-fields.json @@ -75,7 +75,8 @@ { "filter": ["ЧисловоеПоле = 0"], "appearance": { "Текст": { "ru": "Локализованный", "en": "Localized" } } }, { "filter": ["ЧисловоеПоле < 0"], "appearance": { "Текст": "" } }, { "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": "3999-12-31T23:59:59" }], "appearance": { "ЦветТекста": "web:Gray" } }, + { "filter": ["ДатаПоля > 2020-01-01T00:00:00"], "appearance": { "ЦветТекста": "web:Olive" } }, { "filter": [{ "field": "ДатаПоля", "op": ">=", "value": "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 c958e9ca..7a56601f 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 @@ -735,6 +735,25 @@ + + + + + ДатаПоля + Greater + + Custom + 2020-01-01T00:00:00 + + + + + + ЦветТекста + web:Olive + + +