mirror of
https://github.com/Nikolay-Shirokov/cc-1c-skills.git
synced 2026-06-15 02:14:57 +03:00
feat(form-decompile,form-compile): UseAlways — поля реквизита, всегда читаемые (две формы DSL)
<UseAlways><Field>ИмяРеквизита.Поле</Field>…> у Attribute (5189: дин-список 3575 + ValueTable ~788 + прочие) — не захватывался. Свойство «поля, всегда читаемые из БД». DSL — две формы (сливаются компилятором): - на реквизите: useAlways: ["Поле1","Поле2"] (короткие имена; forgiving с/без префикса); - на колонке ValueTable: useAlways: true (columns[*]). Компилятор собирает <Field>ИмяРеквизита.X</Field> из обоих источников (dedupe), порядок схемы: после FillChecking, до FunctionalOptions/Columns/Settings. Дин-список: колонки в XML не эмитятся, но если заданы в DSL с useAlways — формируют UseAlways-массив (Columns подавляются при наличии settings). Декомпилятор по контексту: ValueTable (есть columns) → useAlways:true на совпавшей колонке; дин-список/прочие → массив useAlways на реквизите. Префикс «Имя.» снимается. TOTAL diff lines выборки 2.17: 3869 → 3695 (-174), match 14 → 17 (+3 чистых формы). Attribute>UseAlways residual → 0. Снапшот table (обе формы + merge) сертифицирован в 1С (8.3.24). Регресс form-compile 33/33 зелёный на ps + python. decompile v0.36, compile v1.54. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
@@ -1,4 +1,4 @@
|
||||
# form-compile v1.53 — Compile 1C managed form from JSON or object metadata
|
||||
# form-compile v1.54 — Compile 1C managed form from JSON or object metadata
|
||||
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
|
||||
param(
|
||||
[string]$JsonPath,
|
||||
@@ -3657,10 +3657,36 @@ function Emit-Attributes {
|
||||
if ($attr.fillChecking) {
|
||||
X "$inner<FillChecking>$($attr.fillChecking)</FillChecking>"
|
||||
}
|
||||
|
||||
# UseAlways: поля, всегда читаемые (дин-список/таблица). Две формы DSL сливаются:
|
||||
# attr.useAlways[] (короткие имена) + columns с useAlways:true → <Field>ИмяРеквизита.Поле</Field>.
|
||||
$uaFields = New-Object System.Collections.ArrayList
|
||||
if ($attr.useAlways) {
|
||||
foreach ($e in @($attr.useAlways)) {
|
||||
$fld = "$e"
|
||||
if ($fld -notmatch "^$([regex]::Escape($attrName))\.") { $fld = "$attrName.$fld" }
|
||||
if (-not $uaFields.Contains($fld)) { [void]$uaFields.Add($fld) }
|
||||
}
|
||||
}
|
||||
if ($attr.columns) {
|
||||
foreach ($col in $attr.columns) {
|
||||
if ($col.useAlways -eq $true) {
|
||||
$fld = "$attrName.$($col.name)"
|
||||
if (-not $uaFields.Contains($fld)) { [void]$uaFields.Add($fld) }
|
||||
}
|
||||
}
|
||||
}
|
||||
if ($uaFields.Count -gt 0) {
|
||||
X "$inner<UseAlways>"
|
||||
foreach ($f in $uaFields) { X "$inner`t<Field>$f</Field>" }
|
||||
X "$inner</UseAlways>"
|
||||
}
|
||||
|
||||
Emit-FunctionalOptions -fo $attr.functionalOptions -indent $inner
|
||||
|
||||
# Columns (for ValueTable/ValueTree)
|
||||
if ($attr.columns -and $attr.columns.Count -gt 0) {
|
||||
# Columns (for ValueTable/ValueTree). Для дин-списка (есть settings) колонки НЕ эмитим —
|
||||
# они служат лишь для формирования UseAlways (поля выше).
|
||||
if ($attr.columns -and $attr.columns.Count -gt 0 -and -not $attr.settings) {
|
||||
X "$inner<Columns>"
|
||||
foreach ($col in $attr.columns) {
|
||||
$colId = New-Id
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
#!/usr/bin/env python3
|
||||
# form-compile v1.53 — Compile 1C managed form from JSON or object metadata
|
||||
# form-compile v1.54 — Compile 1C managed form from JSON or object metadata
|
||||
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
|
||||
import argparse
|
||||
import copy
|
||||
@@ -3331,10 +3331,32 @@ def emit_attributes(lines, attrs, indent):
|
||||
lines.append(f'{inner}<SavedData>true</SavedData>')
|
||||
if attr.get('fillChecking'):
|
||||
lines.append(f'{inner}<FillChecking>{attr["fillChecking"]}</FillChecking>')
|
||||
|
||||
# UseAlways: поля, всегда читаемые. Две формы DSL сливаются:
|
||||
# attr.useAlways[] (короткие имена) + columns с useAlways:true → <Field>ИмяРеквизита.Поле</Field>.
|
||||
ua_fields = []
|
||||
for e in (attr.get('useAlways') or []):
|
||||
fld = str(e)
|
||||
if not re.match(r'^' + re.escape(attr_name) + r'\.', fld):
|
||||
fld = f'{attr_name}.{fld}'
|
||||
if fld not in ua_fields:
|
||||
ua_fields.append(fld)
|
||||
for col in (attr.get('columns') or []):
|
||||
if col.get('useAlways') is True:
|
||||
fld = f'{attr_name}.{col["name"]}'
|
||||
if fld not in ua_fields:
|
||||
ua_fields.append(fld)
|
||||
if ua_fields:
|
||||
lines.append(f'{inner}<UseAlways>')
|
||||
for f in ua_fields:
|
||||
lines.append(f'{inner}\t<Field>{f}</Field>')
|
||||
lines.append(f'{inner}</UseAlways>')
|
||||
|
||||
emit_functional_options(lines, attr.get('functionalOptions'), inner)
|
||||
|
||||
# Columns (for ValueTable/ValueTree)
|
||||
if attr.get('columns') and len(attr['columns']) > 0:
|
||||
# Columns (for ValueTable/ValueTree). Для дин-списка (есть settings) колонки НЕ эмитим —
|
||||
# они служат лишь для формирования UseAlways.
|
||||
if attr.get('columns') and len(attr['columns']) > 0 and not attr.get('settings'):
|
||||
lines.append(f'{inner}<Columns>')
|
||||
for col in attr['columns']:
|
||||
col_id = new_id()
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
# form-decompile v0.35 — Decompile 1C managed Form.xml to JSON DSL (draft)
|
||||
# form-decompile v0.36 — Decompile 1C managed Form.xml to JSON DSL (draft)
|
||||
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
|
||||
# ВНИМАНИЕ: раундтрип не гарантируется. Навык исключён из авто-использования моделью.
|
||||
param(
|
||||
@@ -1329,6 +1329,29 @@ if ($attrsNode) {
|
||||
}
|
||||
if ($cols.Count -gt 0) { $ao['columns'] = @($cols) }
|
||||
}
|
||||
# UseAlways: поля, всегда читаемые. Префикс "ИмяРеквизита." снимаем.
|
||||
# ValueTable (есть columns): useAlways:true на совпавшей колонке; остальные → массив атрибута.
|
||||
# Дин-список/прочие (нет columns): массив useAlways на атрибуте.
|
||||
$uaNode = $a.SelectSingleNode("lf:UseAlways", $ns)
|
||||
if ($uaNode) {
|
||||
$prefix = "$($ao['name'])."
|
||||
$shorts = New-Object System.Collections.ArrayList
|
||||
foreach ($fn in @($uaNode.SelectNodes("lf:Field", $ns))) {
|
||||
$t = $fn.InnerText.Trim()
|
||||
if ($t.StartsWith($prefix)) { $t = $t.Substring($prefix.Length) }
|
||||
[void]$shorts.Add($t)
|
||||
}
|
||||
if ($ao.Contains('columns')) {
|
||||
$rest = New-Object System.Collections.ArrayList
|
||||
foreach ($s in $shorts) {
|
||||
$col = $ao['columns'] | Where-Object { $_['name'] -eq $s } | Select-Object -First 1
|
||||
if ($col) { $col['useAlways'] = $true } else { [void]$rest.Add($s) }
|
||||
}
|
||||
if ($rest.Count -gt 0) { $ao['useAlways'] = @($rest) }
|
||||
} elseif ($shorts.Count -gt 0) {
|
||||
$ao['useAlways'] = @($shorts)
|
||||
}
|
||||
}
|
||||
# Settings динамического списка
|
||||
$setNode = $a.SelectSingleNode("lf:Settings", $ns)
|
||||
if ($setNode) {
|
||||
|
||||
@@ -612,6 +612,7 @@ Pages поддерживает `pagesRepresentation`: `None`, `TabsOnTop`, `Tabs
|
||||
| `view` | bool/object | Просмотр по ролям (`<View>`). См. §4.1c |
|
||||
| `edit` | bool/object | Редактирование по ролям (`<Edit>`). См. §4.1c |
|
||||
| `functionalOptions` | array | Функциональные опции (`<FunctionalOptions><Item>FunctionalOption.X</Item>…`). Массив имён; forgiving: `"X"`/`"FunctionalOption.X"`. Также у колонок (`columns[*]`) и команд (§7) |
|
||||
| `useAlways` | array | Поля, всегда читаемые (`<UseAlways><Field>Имя.Поле</Field>…`). Массив коротких имён полей (forgiving: с/без префикса `Имя.`). **Две формы**: этот массив на реквизите ИЛИ `useAlways: true` на колонке (`columns[*]`) — компилятор сливает. Для дин-списка — только массив (колонки не эмитятся, но формируют `<UseAlways>`) |
|
||||
| `savedData` | bool | Сохраняемые данные |
|
||||
| `fillChecking` | string | `Show`, `DontShow` |
|
||||
| `columns` | array | Колонки для ValueTable/ValueTree |
|
||||
|
||||
+4
@@ -106,6 +106,10 @@
|
||||
<Type>
|
||||
<v8:Type>v8:ValueTable</v8:Type>
|
||||
</Type>
|
||||
<UseAlways>
|
||||
<Field>Данные.Сумма</Field>
|
||||
<Field>Данные.Дата</Field>
|
||||
</UseAlways>
|
||||
<Columns>
|
||||
<Column name="Дата" id="31">
|
||||
<Type>
|
||||
|
||||
@@ -35,8 +35,8 @@
|
||||
],
|
||||
"attributes": [
|
||||
{ "name": "Объект", "type": "DataProcessorObject.Таблица", "main": true },
|
||||
{ "name": "Данные", "type": "ValueTable", "columns": [
|
||||
{ "name": "Дата", "type": "date" },
|
||||
{ "name": "Данные", "type": "ValueTable", "useAlways": ["Сумма"], "columns": [
|
||||
{ "name": "Дата", "type": "date", "useAlways": true },
|
||||
{ "name": "Сумма", "type": "decimal(15,2)" },
|
||||
{ "name": "Комментарий", "type": "string(200)" },
|
||||
{ "name": "Объект", "type": "AnyRef" },
|
||||
|
||||
Reference in New Issue
Block a user