sync: UI animations select styling TLS verify flag via proxy second line brand spacing

This commit is contained in:
2025-09-14 11:54:28 +03:00
parent 11a0535712
commit 81014d26f8
11 changed files with 3633 additions and 1459 deletions

View File

@@ -1,3 +1,8 @@
@import url('https://fonts.googleapis.com/css2?family=Merriweather:ital,wght@0,300;0,400;0,700;1,300;1,400;1,700&display=swap');
/* Глобальный шрифт: Merriweather — только для бренда; остальной UI — системный/Inter */
html, body, button, input, select, textarea, code, pre, a, .chip-btn, .group-title, .hint, details.help summary, #sidebar, #inspector, #canvas, #drawflow, .drawflow, .drawflow * {
font-family: Inter, system-ui, Arial, sans-serif;
}
:root {
/* Цвета темы (совпадают с editor.html) */
color-scheme: dark;
@@ -13,6 +18,53 @@
--connector: #7aa2f7;
--connector-muted: #3b82f6;
}
html, body {
height: 100%;
overflow: hidden; /* убираем общие скролл-бары страницы, чтобы не перекрывать правую стрелку */
}
/* Глобальные контейнеры и скроллы */
html, body {
height: 100%;
overflow: hidden; /* убираем общие скролл-бары страницы */
}
#container {
position: relative; /* якорь для абсолютных стрелок-переключателей */
}
/* Глобальные контейнеры и скроллы */
html, body {
height: 100%;
overflow: hidden; /* убираем общие скролл-бары страницы */
}
#container {
position: relative; /* якорь для абсолютных стрелок-переключателей */
}
/* Grid areas to hard-pin layout regardless of hidden panels or absolute children */
#container {
display: grid;
grid-template-areas: "side main insp";
/* Базовая ширина колонок по умолчанию; переопределяется collapse-* ниже */
grid-template-columns: 260px 1fr 360px;
/* Растяжение по высоте: родитель (body/html) уже 100% */
height: 100%;
min-height: 100%;
}
/* Map children to areas explicitly */
#container > #sidebar { grid-area: side; }
#container > #canvas { grid-area: main; }
#container > #inspector { grid-area: insp; }
/* Снимаем скролл-бары с контейнера Drawflow, чтобы не перекрывать правую стрелку */
#drawflow {
overflow: hidden !important;
position: relative;
z-index: 1; /* гарантируем, что канвас виден под HUD и над фоном */
/* Растянем контейнер Drawflow на всю центральную колонку */
width: 100%;
height: 100%;
display: block;
}
/* Узлы: аккуратные контейнеры + предотвращение вылезания текста */
.drawflow .drawflow-node {
@@ -36,6 +88,11 @@
border-radius: 0 0 12px 12px;
overflow: hidden; /* не даём контенту вылезать за края */
}
/* Контент превью внутри .box: можем скрывать его в LOD, не ломая геометрию ноды */
.drawflow .drawflow-node .node-preview {
pointer-events: none;
opacity: .85;
}
.drawflow .drawflow-node .box textarea,
.drawflow .drawflow-node .box pre,
@@ -52,8 +109,9 @@
.df-node .box textarea {
white-space: pre-wrap;
word-break: break-word;
overflow: auto;
max-height: 180px; /* предотвращаем бесконечную высоту */
overflow-y: auto; /* только вертикальный скролл при необходимости */
overflow-x: hidden; /* убираем горизонтальный скролл внутри textarea */
max-height: 180px; /* предотвращаем бесконечную высоту */
}
/* Выделение выбранного узла — мягкое */
@@ -68,22 +126,24 @@
.drawflow .drawflow-node .outputs .output {
background: var(--accent-2) !important;
border: 2px solid color-mix(in srgb, var(--accent-2) 70%, white 0%) !important;
width: 12px !important;
height: 12px !important;
width: 16px !important; /* ↑ hit-area */
height: 16px !important; /* ↑ hit-area */
box-shadow: 0 0 0 2px rgba(0,0,0,.25);
}
/* Линии соединений: плавные, аккуратные цвета */
.drawflow .connection .main-path {
stroke: var(--connector) !important;
stroke-width: 2.5px !important;
/* Толщина линии масштабируется от зума (var(--zoom) задаётся на #canvas из JS) */
stroke-width: clamp(1.6px, calc(3px / var(--zoom, 1)), 6px) !important;
opacity: 0.95 !important;
}
.drawflow .connection .main-path.selected,
.drawflow .connection:hover .main-path {
stroke: var(--accent-2) !important;
stroke-width: 3px !important;
/* На hover/selected — немного толще базовой формулы */
stroke-width: clamp(2px, calc(3.6px / var(--zoom, 1)), 7px) !important;
}
/* Точки изгибов/ручки */
@@ -139,6 +199,22 @@ details.help .panel {
border-radius: 8px;
}
/* Sidebar help: make the toggle text styled like regular sidebar hint/title, not a round icon */
#sidebar details.sidebar-help summary.sidebar-help-toggle {
all: unset; /* сбросить круглую кнопку от общего ruleset */
cursor: pointer;
color: var(--muted);
font-size: 12px;
line-height: 1.4;
display: block; /* как обычный текст строки подсказки */
margin: 4px 0 6px; /* как у .hint */
}
#sidebar details.sidebar-help summary.sidebar-help-toggle:hover {
color: #cbd5e1;
text-decoration: underline;
}
#sidebar details.sidebar-help summary.sidebar-help-toggle::-webkit-details-marker { display: none; }
/* Инпуты/тексты внутри нод — одинаковые отступы и скругления */
textarea, input[type=text] {
width: 100%;
@@ -149,7 +225,7 @@ textarea, input[type=text] {
padding: 8px;
}
/* Кнопки */
/* Кнопки (базовые) */
button {
background: #1f2937;
border: 1px solid #334155;
@@ -160,6 +236,70 @@ button {
}
button:hover { background: #273246; }
/* Верхняя панель — стиль как у чип‑стрелок и как заголовок «ПЕРЕМЕННЫЕ И МАКРОСЫ» */
header .brand {
font-size: 12px;
text-transform: uppercase;
color: var(--muted);
letter-spacing: .08em;
font-weight: 700;
font-family: 'Merriweather', serif !important;
}
header .actions {
display: flex;
flex-wrap: wrap;
gap: 8px;
align-items: center;
}
/* Чип‑кнопки в шапке (и ссылки как кнопки) */
.chip-btn,
a.chip-btn {
display: inline-grid;
place-items: center;
text-decoration: none;
text-transform: uppercase;
font-size: 12px;
letter-spacing: .04em;
padding: 6px 10px;
border-radius: 8px;
background: #0f141a;
color: #e5e7eb;
border: 1px solid #334155;
box-shadow: 0 2px 6px rgba(0,0,0,.35);
transition: transform .12s ease, box-shadow .12s ease, background-color .12s ease, border-color .12s ease, color .12s ease;
user-select: none;
}
.chip-btn:hover,
a.chip-btn:hover {
background: #1f2937;
border-color: var(--accent-2);
box-shadow: 0 0 0 3px rgba(96,165,250,.22), 0 4px 10px rgba(0,0,0,.35);
}
.chip-btn:active,
a.chip-btn:active {
transform: translateY(1px);
box-shadow: 0 0 0 2px rgba(96,165,250,.20), 0 2px 6px rgba(0,0,0,.35);
}
/* Инпуты и селекты в шапке — в одном визуальном ряду с чипами */
.top-input {
height: 32px;
min-width: 160px;
background: #0f141a;
border: 1px solid #334155;
border-radius: 8px;
color: #e5e7eb;
padding: 6px 8px;
box-sizing: border-box;
outline: none;
}
.top-input:focus {
border-color: var(--accent-2);
box-shadow: 0 0 0 3px rgba(96,165,250,.20);
}
/* Внутренние заголовки в блоке ноды */
#inspector label { font-size: 12px; color: var(--muted); display: block; margin: 8px 0 4px; }
@@ -219,6 +359,327 @@ button:hover { background: #273246; }
background-color: var(--bg);
background-image: radial-gradient(circle at 1px 1px, rgba(255,255,255,0.06) 1px, transparent 0);
background-size: 24px 24px;
position: relative; /* ensure HUD overlay absolute positioning works */
overflow: visible; /* позволяем HUD-элементам (чипам) выходить за границы канваса */
/* Растяжение на всю область grid-ячейки main */
width: 100%;
height: 100%;
display: block;
pointer-events: auto;
}
/* HUD overlay layer for LOD labels (screen-space, not affecting node geometry) */
#lod-label-layer,
#lod-hints {
position: absolute;
inset: 0;
pointer-events: none; /* do not intercept mouse */
z-index: 5; /* над нодами/линиями (у них z-index auto), но ниже боковых панелей */
overflow: visible; /* НЕ клипать чипы — пусть уходят под панели */
}
/* Visual chip for node label at far zoom */
/* Generic LOD chip style (used in #lod-hints and legacy #lod-label-layer) */
.lod-chip {
position: absolute;
background: #10151c;
color: #e5e7eb;
border: 2px solid #3b82f6;
border-radius: 8px;
padding: 2px 6px;
white-space: nowrap;
line-height: 1.2;
text-shadow: 0 1px 1px rgba(0,0,0,.6);
box-shadow: 0 0 0 2px rgba(59,130,246,.20), 0 2px 6px rgba(0,0,0,.35);
transform: translate(-50%, -100%); /* centered by JS; here as default */
font: 18px/1.2 Inter, system-ui, Arial, sans-serif;
}
/* LOD: level-of-detail tuned by classes on #canvas (set from JS) */
#canvas.lod-compact .drawflow .drawflow-node .node-preview,
#canvas.lod-tiny .drawflow .drawflow-node .node-preview {
display: none !important; /* скрываем только содержимое превью, не меняя коробку ноды */
}
/* Порты и линии — тоньше в компактных режимах */
/* Убраны фиксированные толщины в LOD — используется формула от --zoom */
#canvas.lod-compact .drawflow .drawflow-node .inputs .input,
#canvas.lod-compact .drawflow .drawflow-node .outputs .output { width: 12px !important; height: 12px !important; }
#canvas.lod-tiny .drawflow .drawflow-node .inputs .input,
#canvas.lod-tiny .drawflow .drawflow-node .outputs .output { width: 10px !important; height: 10px !important; }
/* Индикатор статуса — свечением по рамке остаётся, доп. эффект на тайтле */
#canvas.lod-compact .drawflow .drawflow-node.node-running .title-box,
#canvas.lod-tiny .drawflow .drawflow-node.node-running .title-box { box-shadow: 0 0 0 2px rgba(96,165,250,.25) !important; }
#canvas.lod-compact .drawflow .drawflow-node.node-ok .title-box,
#canvas.lod-tiny .drawflow .drawflow-node.node-ok .title-box { box-shadow: 0 0 0 2px rgba(52,211,153,.25) !important; }
#canvas.lod-compact .drawflow .drawflow-node.node-err .title-box,
#canvas.lod-tiny .drawflow .drawflow-node.node-err .title-box { box-shadow: 0 0 0 2px rgba(239,68,68,.25) !important; }
/* Sleep (amber) */
#canvas.lod-compact .drawflow .drawflow-node.node-sleep .title-box,
#canvas.lod-tiny .drawflow .drawflow-node.node-sleep .title-box { box-shadow: 0 0 0 2px rgba(245,158,11,.28) !important; }
/* Run drawer (settings) */
#run-drawer {
position: fixed;
top: 52px;
right: -380px;
width: 360px;
height: calc(100vh - 52px);
background: var(--panel);
border-left: 1px solid var(--border);
box-shadow: -8px 0 24px rgba(0,0,0,.35);
z-index: 10000;
transition: right .18s ease-in-out;
padding: 12px;
color: #e5e7eb;
}
#run-drawer.open { right: 0; }
#run-drawer h3 { margin: 0 0 8px; font-size: 14px; color: var(--muted); }
#run-drawer .row { display: flex; gap: 8px; margin: 8px 0; align-items: center; }
#run-drawer input[type=number], #run-drawer select {
width: 100%;
background: #0f141a;
border: 1px solid #2b3646;
color: #e5e7eb;
border-radius: 8px;
padding: 6px 8px;
}
#run-drawer .actions { display: flex; justify-content: space-between; gap: 8px; margin-top: 12px; }
/* Logs panel — перенос строк и стили списка + подсветка видов событий */
#logs-panel pre {
white-space: pre-wrap;
word-break: break-word;
overflow-wrap: anywhere;
}
/* Список логов слева */
#logs-panel #logs-list {
overflow-y: auto;
overflow-x: hidden;
}
#logs-panel #logs-list .logs-row {
padding: 8px 10px;
border-bottom: 1px solid #1f2b3b;
cursor: pointer;
display: block;
}
#logs-panel #logs-list .logs-row .title {
font-weight: 600;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
#logs-panel #logs-list .logs-row .sub {
opacity: .85;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
#logs-panel #logs-list .logs-row:hover {
background: rgba(96,165,250,.08);
}
#logs-panel #logs-list .logs-row.selected {
background: #0f1e2d;
outline: 2px solid #60a5fa;
box-shadow: 0 0 0 2px rgba(96,165,250,.18) inset;
}
#logs-panel #logs-list .logs-row.selected .title {
color: #eaf2ff;
}
/* Dim non-main logs (anything except start/done/error and HTTP errors) */
#logs-panel #logs-list .logs-row.dim:not(.selected) {
opacity: .55;
}
#logs-panel #logs-list .logs-row.dim:not(.selected) .title {
color: #94a3b8;
font-weight: 500;
}
#logs-panel #logs-list .logs-row.dim:not(.selected) .sub {
color: #7c8797;
}
/* Ensure selection overrides dimming */
#logs-panel #logs-list .logs-row.selected.dim {
opacity: 1 !important;
}
/* Subtle emphasis for main node events even when not selected */
#logs-panel #logs-list .logs-row.kind-node.ev-start:not(.selected) {
background: rgba(96,165,250,.07);
}
#logs-panel #logs-list .logs-row.kind-node.ev-done:not(.selected) {
background: rgba(52,211,153,.07);
}
/* Типы логов — цветовая метка по левому бордеру */
#logs-panel #logs-list .logs-row.kind-node.ev-start { border-left: 3px solid #60a5fa; } /* start — синий */
#logs-panel #logs-list .logs-row.kind-node.ev-done { border-left: 3px solid #34d399; } /* done — зелёный */
#logs-panel #logs-list .logs-row.kind-node.ev-error { border-left: 3px solid #ef4444; } /* error — красный */
#logs-panel #logs-list .logs-row.kind-node.ev-sleep { border-left: 3px solid #f59e0b; } /* sleep — янтарный */
#logs-panel #logs-list .logs-row.kind-http.http-ok { border-left: 3px solid #34d399; } /* HTTP 2xx/3xx */
#logs-panel #logs-list .logs-row.kind-http.http-err { border-left: 3px solid #ef4444; } /* HTTP 4xx/5xx */
#logs-panel #logs-list .logs-row.kind-vars { border-left: 3px solid #a78bfa; } /* Vars — фиолетовый */
/* Тонкие скролл-бары для панелей/textarea, чтобы не «мешали» интерфейсу */
#sidebar, #inspector, #scheme-panel, #run-drawer, .vars-popover, #logs-panel pre, #logs-list {
scrollbar-width: thin;
scrollbar-color: #334155 transparent;
}
#sidebar::-webkit-scrollbar,
#inspector::-webkit-scrollbar,
#scheme-panel::-webkit-scrollbar,
#run-drawer::-webkit-scrollbar,
.vars-popover::-webkit-scrollbar,
#logs-panel pre::-webkit-scrollbar,
#logs-list::-webkit-scrollbar {
width: 8px;
height: 8px;
}
#sidebar::-webkit-scrollbar-thumb,
#inspector::-webkit-scrollbar-thumb,
#scheme-panel::-webkit-scrollbar-thumb,
#run-drawer::-webkit-scrollbar-thumb,
.vars-popover::-webkit-scrollbar-thumb,
#logs-panel pre::-webkit-scrollbar-thumb,
#logs-list::-webkit-scrollbar-thumb {
background-color: #334155;
border-radius: 8px;
}
/* Убираем горизонтальный скролл в текстовых редакторах инспектора */
#inspector textarea { overflow-x: hidden; }
/* Mini-scheme panel */
#scheme-panel {
position: fixed;
top: 52px;
right: 24px;
width: 380px;
height: auto;
max-height: calc(100vh - 72px);
background: var(--panel);
border: 1px solid var(--border);
box-shadow: -8px 0 24px rgba(0,0,0,.35);
z-index: 10000;
padding: 12px;
color: #e5e7eb;
overflow: hidden; /* запрещаем выход содержимого за границы панели */
}
#scheme-panel h3 { margin: 0 0 8px; font-size: 14px; color: var(--muted); }
#scheme-panel canvas {
width: 100%;
height: auto; /* фактический размер управляется JS с учётом DPR */
border: 1px solid #2b3646;
border-radius: 8px;
background: #0f141a;
display: block; /* убираем нижний пробел как у inline */
}
/* Docked STORE panel (reuse existing popover node, просто фиксируем сбоку) */
#vars-popover {
position: fixed !important;
top: 52px !important;
right: 24px !important;
height: calc(100vh - 72px) !important;
max-height: none !important;
z-index: 9999 !important;
}
/* Vars popover visual layout */
.vars-popover {
width: 520px; /* фиксированная удобная ширина */
background: #0f141a;
border: 1px solid #2b3646;
border-radius: 10px;
box-shadow: 0 6px 28px rgba(0,0,0,.45);
color: #e5e7eb;
overflow: hidden; /* чтобы шапка/низ не вылезали */
}
/* Header as compact grid */
.vars-popover .vars-head {
display: grid;
grid-template-columns: 1fr minmax(160px, 1.2fr) 120px auto auto auto;
gap: 8px;
align-items: center;
padding: 10px;
border-bottom: 1px solid #2b3646;
}
.vars-popover .vars-head > strong {
font-size: 14px;
color: var(--muted);
}
.vars-popover .vars-head input#vars-search,
.vars-popover .vars-head select#vars-scope {
width: 100%;
background: #0f141a;
border: 1px solid #2b3646;
color: #e5e7eb;
border-radius: 8px;
padding: 6px 8px;
box-sizing: border-box;
}
.vars-popover .vars-head .vars-braces {
display: inline-flex;
align-items: center;
gap: 6px;
font-size: 12px;
color: #a7b0bf;
white-space: nowrap;
}
.vars-popover .vars-head button {
background: #1f2937;
border: 1px solid #334155;
color: #e5e7eb;
padding: 6px 8px;
border-radius: 8px;
cursor: pointer;
}
.vars-popover .vars-head button:hover { background: #273246; }
/* Info line */
.vars-popover #vars-info {
padding: 8px 10px;
border-bottom: 1px solid #2b3646;
color: #a7b0bf;
}
/* List area */
.vars-popover #vars-list {
padding: 8px 0;
overflow: auto;
max-height: calc(100% - 110px); /* остаток высоты относительно контейнера */
}
/* Rows styling (override inline minimal grid by sizing columns and clamping) */
.vars-popover #vars-list .row {
display: grid !important;
grid-template-columns: 220px 1fr !important; /* фиксируем ширину колонки макроса */
gap: 8px !important;
padding: 8px 10px !important;
border-bottom: 1px solid #1f2b3b !important;
cursor: pointer !important;
}
.vars-popover #vars-list .row:hover {
background: rgba(96,165,250,.08);
}
.vars-popover #vars-list .row code {
font-family: ui-monospace, SFMono-Regular, Menlo, Consolas, "Liberation Mono", monospace;
color: #60a5fa !important;
overflow: hidden !important;
text-overflow: ellipsis !important;
white-space: nowrap !important;
max-width: 220px !important;
display: block !important;
}
.vars-popover #vars-list .row > div {
overflow: hidden !important;
text-overflow: ellipsis !important;
white-space: nowrap !important;
}
/* Port hover affordance (no heavy effects) */
@@ -324,4 +785,304 @@ button:hover { background: #273246; }
.drawflow .drawflow-node.node-err .box {
border-color: #ef4444 !important; /* red */
box-shadow: 0 0 0 2px rgba(239,68,68,.35) !important;
}
/* Sleep state — amber */
.drawflow .drawflow-node.node-sleep .title-box,
.drawflow .drawflow-node.node-sleep .box {
border-color: #f59e0b !important; /* amber */
box-shadow: 0 0 0 2px rgba(245,158,11,.35) !important;
}
/* LOD hover tooltip (shown only in compact/tiny via JS) */
.lod-tooltip {
position: absolute;
pointer-events: none;
background: #10151c;
color: #e5e7eb;
border: 2px solid #3b82f6;
border-radius: 8px;
padding: 2px 6px;
white-space: nowrap;
font: 18px/1.2 Inter, system-ui, Arial, sans-serif;
box-shadow: 0 0 0 2px rgba(59,130,246,.20), 0 2px 6px rgba(0,0,0,.35);
transform: translate(-50%, -100%);
z-index: 1000; /* above nodes/edges but below menus */
}
/* Снимаем скролл-бары с контейнера Drawflow, чтобы не перекрывать правую стрелку */
#drawflow {
overflow: hidden !important;
position: relative;
z-index: 1; /* гарантируем, что канвас виден под HUD и над фоном */
/* Растянем контейнер Drawflow на всю центральную колонку */
width: 100%;
height: 100%;
display: block;
}
/* Panels collapse controls and layout */
#container.collapse-left {
grid-template-columns: 0 1fr 360px !important;
}
#container.collapse-right {
grid-template-columns: 260px 1fr 0 !important;
}
/* BOTH collapsed: высокий приоритет, чтобы не осталась «пустая колонка» */
#container.collapse-left.collapse-right {
grid-template-columns: 0 1fr 0 !important;
}
/* removed explicit grid placement to avoid grid misplacement on some browsers */
/* Поверх HUD: панели выше, чтобы чипы уходили ПОД них */
#sidebar, #inspector {
position: relative;
z-index: 10;
}
/* Inspector: extra bottom padding + no horizontal scroll to ensure OUTx block reachable */
#inspector {
overflow-y: auto;
overflow-x: hidden;
padding-bottom: 240px;
}
#container.collapse-left #sidebar {
/* Полностью исключаем сайдбар из потока при сворачивании,
чтобы он не создавал «чёрную накладку» и не перехватывал клики */
display: none !important;
/* Страховочные свойства на случай кеша/старых стилей */
width: 0 !important;
padding: 0 !important;
border-right: 0 !important;
overflow: hidden !important;
pointer-events: none !important;
background: transparent !important;
visibility: hidden !important;
}
#container.collapse-right #inspector {
/* Полностью исключаем инспектор из потока и стэка,
чтобы гарантированно не перекрывал канвас (фикс «чёрной накладки») */
display: none !important;
/* Ниже — страховочные свойства для старых кешированных стилей */
width: 0 !important;
padding: 0 !important;
border-left: 0 !important;
overflow: hidden !important;
pointer-events: none !important;
background: transparent !important;
visibility: hidden !important;
}
/* На случай, если браузер продолжит хит‑тестить потомков —
дополнительно отключаем события на всём поддереве инспектора */
#container.collapse-right #inspector * {
pointer-events: none !important;
}
/* Edge toggle buttons — positioned relative to #container; coordinates set by JS (placeToggles) */
.panel-toggle {
position: absolute;
top: 50%;
transform: translateY(-50%); /* vertical center */
width: 22px;
height: 36px;
display: grid;
place-items: center;
border-radius: 8px;
background: #0f141a;
color: #e5e7eb;
border: 1px solid #334155;
box-shadow: 0 2px 6px rgba(0,0,0,.35);
z-index: 9000; /* below drawers/popovers (#run-drawer/#scheme-panel/#vars-popover), above HUD/chips */
cursor: pointer;
user-select: none;
opacity: .95;
pointer-events: auto; /* ensure clickable over HUD layers */
}
/* Left/Right offsets are set dynamically via JS, no static left/right here */
.panel-toggle:hover {
background: #1f2937;
border-color: var(--accent-2);
box-shadow: 0 0 0 3px rgba(96,165,250,.22), 0 4px 10px rgba(0,0,0,.35);
}
/* Safety: hide any leftover debug overlay if it still exists in DOM */
#dbg-hit { display: none !important; }
/* Remove dotted/dashed focus outlines on controls (requested) */
button:focus,
a.chip-btn:focus,
.chip-btn:focus,
.panel-toggle:focus {
outline: none !important;
box-shadow: none !important;
}
/* Logs UI: stronger selected row and readable details override */
#logs-panel #logs-list .logs-row.selected {
border-left-width: 5px; /* stronger accent on selected */
}
#logs-panel #logs-list .logs-row.selected .title {
font-weight: 700;
letter-spacing: .01em;
}
#logs-panel #logs-list .logs-row.selected .sub {
color: #cbd5e1;
}
/* Smooth collapse/expand of side panels via grid columns transition */
#container {
transition: grid-template-columns .18s ease-in-out;
}
/* Optional: soften perceived change on panel boxes themselves */
#sidebar,
#inspector {
transition: opacity .18s ease-in-out;
}
#container.collapse-left #sidebar { opacity: 0; }
#container.collapse-right #inspector { opacity: 0; }
/* Smooth appear animations for overlay/panels (trigger on display: block) */
@keyframes fadeInUp {
from { opacity: 0; transform: translateY(6px); }
to { opacity: 1; transform: translateY(0); }
}
@keyframes fadeInRight {
from { opacity: 0; transform: translateX(8px); }
to { opacity: 1; transform: translateX(0); }
}
@keyframes fadeInDown {
from { opacity: 0; transform: translateY(-6px); }
to { opacity: 1; transform: translateY(0); }
}
/* Respect reduced motion */
@media (prefers-reduced-motion: reduce) {
#container {
transition: none !important;
}
#sidebar, #inspector,
#vars-popover, #logs-panel, #scheme-panel {
transition: none !important;
animation: none !important;
}
}
/* Vars popover: quick fade + slight upward motion on open (display becomes block in JS) */
#vars-popover[style*="display: block"] {
animation: fadeInUp .16s ease-out;
will-change: opacity, transform;
}
/* Logs panel: fade + slight leftward motion (panel is fixed on the right) */
#logs-panel[style*="display: block"] {
animation: fadeInRight .16s ease-out;
will-change: opacity, transform;
}
/* Mini-scheme panel: fade + slight downward motion (drops from header area) */
#scheme-panel[style*="display: block"] {
animation: fadeInDown .16s ease-out;
will-change: opacity, transform;
}
/* Unify select look to match preset-select (header .top-input style) */
select#f-provider,
select#tx-preset,
select#vars-scope,
select.v-mode,
select#pm-role {
appearance: none;
-webkit-appearance: none;
-moz-appearance: none;
background: #0f141a;
color: #e5e7eb;
border: 1px solid #334155;
border-radius: 8px;
height: 32px;
padding: 6px 30px 6px 8px;
box-sizing: border-box;
outline: none;
font: 12px/1 Inter, system-ui, Arial, sans-serif;
transition:
border-color .12s ease,
box-shadow .12s ease,
background-color .12s ease,
color .12s ease;
}
/* Hover and focus states consistent with .top-input */
select#f-provider:hover,
select#tx-preset:hover,
select#vars-scope:hover,
select.v-mode:hover,
select#pm-role:hover {
background: #121820;
border-color: var(--accent-2);
}
select#f-provider:focus,
select#tx-preset:focus,
select#vars-scope:focus,
select.v-mode:focus,
select#pm-role:focus {
border-color: var(--accent-2);
box-shadow: 0 0 0 3px rgba(96,165,250,.20);
}
/* Compact width contexts: keep natural width unless container forces 100% */
#inspector select#f-provider,
#inspector select#tx-preset,
#inspector select.v-mode,
#inspector select#pm-role {
width: 100%;
}
/* Vars dock header already grid-fit; just unify visuals */
.vars-popover .vars-head select#vars-scope {
height: 32px;
}
/* Add a minimal dropdown arrow via background SVG, aligned right */
select#f-provider,
select#tx-preset,
select#vars-scope,
select.v-mode,
select#pm-role {
background-image:
url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 24 24' fill='none' stroke='%23a7b0bf' stroke-width='2' stroke-linecap='round' stroke-linejoin='round'><polyline points='6 9 12 15 18 9'/></svg>");
background-repeat: no-repeat;
background-position: right 8px center;
background-size: 12px 12px;
}
/* High-contrast for disabled */
select#f-provider:disabled,
select#tx-preset:disabled,
select#vars-scope:disabled,
select.v-mode:disabled,
select#pm-role:disabled {
opacity: .6;
cursor: not-allowed;
}
/* Respect reduced motion for focus rings */
@media (prefers-reduced-motion: reduce) {
select#f-provider,
select#tx-preset,
select#vars-scope,
select.v-mode,
select#pm-role {
transition: none;
}
}
/* Header spacing: add safe gap after brand and small indent for wrapped action rows */
header .brand {
margin-right: 12px; /* space after "НАДTAVERN" */
padding-right: 2px; /* micro buffer to avoid optical collision */
}
header .actions {
margin-left: 8px; /* indent when rows wrap under brand */
}
@media (max-width: 860px) {
header .brand { margin-right: 10px; }
header .actions { margin-left: 6px; }
}

File diff suppressed because it is too large Load Diff

View File

@@ -20,6 +20,18 @@
loop_max_iters: 1000,
loop_time_budget_ms: 10000,
clear_var_store: true,
// New: default HTTP timeout for upstream requests (seconds)
http_timeout_sec: 60,
// New (v1): стратегия извлечения текста для [[OUTx]] (глобальная по умолчанию)
// auto | deep | openai | gemini | claude | jsonpath
text_extract_strategy: 'auto',
// Используется при стратегии jsonpath (dot-нотация, поддержка индексов: a.b.0.c)
text_extract_json_path: '',
// Разделитель при объединении массива результатов
text_join_sep: '\n',
// v2: коллекция пресетов извлечения текста, управляется в "Запуск"
// [{ id, name, strategy, json_path, join_sep }]
text_extract_presets: [],
};
function getPipelineMeta() {
@@ -28,16 +40,45 @@
function updatePipelineMeta(p) {
if (!p || typeof p !== 'object') return;
const keys = ['id','name','parallel_limit','loop_mode','loop_max_iters','loop_time_budget_ms','clear_var_store'];
const keys = [
'id','name','parallel_limit','loop_mode','loop_max_iters','loop_time_budget_ms','clear_var_store','http_timeout_sec',
'text_extract_strategy','text_extract_json_path','text_join_sep','text_join_sep','text_join_SEP',
// v2 presets collection
'text_extract_presets'
];
for (const k of keys) {
if (Object.prototype.hasOwnProperty.call(p, k) && p[k] !== undefined && p[k] !== null && (k === 'clear_var_store' ? true : p[k] !== '')) {
if (k === 'parallel_limit' || k === 'loop_max_iters' || k === 'loop_time_budget_ms') {
const v = parseInt(p[k], 10);
if (!Number.isNaN(v) && v > 0) _pipelineMeta[k] = v;
} else if (k === 'http_timeout_sec') {
const fv = parseFloat(p[k]);
if (!Number.isNaN(fv) && fv > 0) _pipelineMeta[k] = fv;
} else if (k === 'clear_var_store') {
_pipelineMeta[k] = !!p[k];
} else {
_pipelineMeta[k] = String(p[k]);
// спец-обработка коллекции пресетов
if (k === 'text_extract_presets') {
try {
const arr = Array.isArray(p[k]) ? p[k] : [];
_pipelineMeta[k] = arr
.filter(it => it && typeof it === 'object')
.map(it => ({
id: String((it.id ?? '') || ('p' + Date.now().toString(36) + Math.random().toString(36).slice(2))),
name: String(it.name ?? 'Preset'),
strategy: String(it.strategy ?? 'auto'),
json_path: String(it.json_path ?? ''),
join_sep: String(it.join_sep ?? '\n'),
}));
} catch (_) {
_pipelineMeta[k] = [];
}
} else if (k.toLowerCase() === 'text_join_sep') {
// нормализация ключа join separator (допускаем разные написания)
_pipelineMeta['text_join_sep'] = String(p[k]);
} else {
_pipelineMeta[k] = String(p[k]);
}
}
}
}
@@ -60,11 +101,24 @@
const wantIds = {}; // drawflow id -> желаемый/финальный nX
const isValidNid = (s) => typeof s === 'string' && /^n\d+$/i.test(s.trim());
// Helper: вернуть исключительно «живые» данные ноды из редактора (если доступны).
// Это исключает расхождения между DOM.__data и editor.getNodeFromId(..).data.
function mergedNodeData(df, el, dfid) {
try {
const nid = parseInt(dfid, 10);
const n = (w.editor && typeof w.editor.getNodeFromId === 'function') ? w.editor.getNodeFromId(nid) : null;
if (n && n.data) return n.data;
} catch (_) {}
if (df && df.data) return df.data;
// как последний fallback — DOM.__data (почти не используется после этого изменения)
return (el && el.__data) ? el.__data : {};
}
// Первый проход: резервируем существующие валидные _origId
for (const id in dfNodes) {
const df = dfNodes[id];
const el = document.querySelector(`#node-${id}`);
const datacopySrc = el && el.__data ? el.__data : (df.data || {});
const datacopySrc = mergedNodeData(df, el, id);
const tmp = typeof w.applyNodeDefaults === 'function'
? w.applyNodeDefaults(df.name, JSON.parse(JSON.stringify(datacopySrc)))
: (JSON.parse(JSON.stringify(datacopySrc)));
@@ -95,11 +149,22 @@
for (const id in dfNodes) {
const df = dfNodes[id];
const el = document.querySelector(`#node-${id}`);
const datacopySrc = el && el.__data ? el.__data : (df.data || {});
const datacopySrc = mergedNodeData(df, el, id);
const datacopy = typeof w.applyNodeDefaults === 'function'
? w.applyNodeDefaults(df.name, JSON.parse(JSON.stringify(datacopySrc)))
: (JSON.parse(JSON.stringify(datacopySrc)));
try { datacopy._origId = idMap[id]; } catch (e) {}
// Спец-обработка SetVars: гарантированно берём свежие variables из живых данных редактора
try {
if (String(df.name) === 'SetVars') {
const nid = parseInt(id, 10);
const nLive = (w.editor && typeof w.editor.getNodeFromId === 'function') ? w.editor.getNodeFromId(nid) : null;
const v = nLive && nLive.data && Array.isArray(nLive.data.variables) ? nLive.data.variables : (Array.isArray(datacopy.variables) ? datacopy.variables : []);
datacopy.variables = v.map(x => ({ ...(x || {}) })); // глубокая копия
}
} catch (_) {}
nodes.push({
id: idMap[id],
type: df.name,
@@ -205,6 +270,12 @@
loop_max_iters: (typeof meta.loop_max_iters === 'number' ? meta.loop_max_iters : 1000),
loop_time_budget_ms: (typeof meta.loop_time_budget_ms === 'number' ? meta.loop_time_budget_ms : 10000),
clear_var_store: (typeof meta.clear_var_store === 'boolean' ? meta.clear_var_store : true),
http_timeout_sec: (typeof meta.http_timeout_sec === 'number' ? meta.http_timeout_sec : 60),
text_extract_strategy: (meta.text_extract_strategy || 'auto'),
text_extract_json_path: (meta.text_extract_json_path || ''),
text_join_sep: (meta.text_join_sep || '\n'),
// v2: persist presets
text_extract_presets: (Array.isArray(meta.text_extract_presets) ? meta.text_extract_presets : []),
nodes
};
}
@@ -225,6 +296,12 @@
loop_max_iters: (p && typeof p.loop_max_iters === 'number') ? p.loop_max_iters : 1000,
loop_time_budget_ms: (p && typeof p.loop_time_budget_ms === 'number') ? p.loop_time_budget_ms : 10000,
clear_var_store: (p && typeof p.clear_var_store === 'boolean') ? p.clear_var_store : true,
http_timeout_sec: (p && typeof p.http_timeout_sec === 'number') ? p.http_timeout_sec : 60,
text_extract_strategy: (p && typeof p.text_extract_strategy === 'string') ? p.text_extract_strategy : 'auto',
text_extract_json_path: (p && typeof p.text_extract_json_path === 'string') ? p.text_extract_json_path : '',
text_join_sep: (p && typeof p.text_join_sep === 'string') ? p.text_join_sep : '\n',
// v2: presets from pipeline.json
text_extract_presets: (p && Array.isArray(p.text_extract_presets)) ? p.text_extract_presets : [],
});
} catch (e) {}