Files
invyone/frontend/styles/control-mode.css
T
DDD1542 2f398ae0b3 chore: 제어모드 IDE 작업 + v2/legacy 레지스트리 컴포넌트 폐기
- 제어모드 IDE: ControlCardPanel, control/ide/* (Canvas/LeftRail/RightRail/PanZoomStage/V3RuleNode 등), schemas, lib/api/control
- 레지스트리 정리: aggregation-widget, status-count, section-card/paper, table-list(legacy/v2), tabs-widget 폐기 → table/_shared/ 로 통합
- InvLegacyButtonConfigPanel cp 마이그레이션
- canonical data view cleanup 후속 노트
2026-05-19 21:31:03 +09:00

2606 lines
84 KiB
CSS

/* ═══════════════════════════════════════════════════════════════════════════
Phase 5 — 제어 모드 + 규칙 빌더
mockup css/07-control-mode.css + css/08-rule-builder.css → React 포팅
★ --v5-* / --ctrl-* 변수 사용, 즉흥 hex 금지
═══════════════════════════════════════════════════════════════════════════ */
/* ── 제어 모드 변수 ── */
:root {
--ctrl-cyan: #00cec9;
--ctrl-cyan-glow: rgba(var(--v5-cyan-rgb), .3);
--ctrl-primary: #6c5ce7;
--ctrl-amber: #fdcb6e;
--ctrl-pink: #fd79a8;
--ctrl-green: #55efc4;
--ctrl-red: #ff4757;
--ctrl-glass: rgba(255, 255, 255, .06);
--ctrl-glass-strong: rgba(255, 255, 255, .08);
--ctrl-glass-border: rgba(var(--v5-cyan-rgb), .25);
}
.dark {
--ctrl-glass: rgba(255, 255, 255, .04);
--ctrl-glass-strong: rgba(255, 255, 255, .06);
}
/* ═══ 편집/제어 모드 캔버스 — INVYONE Design System (ui_kits/app) 포팅 ═══
primary/cyan tint + 24px line grid + inset 2px border.
기본 dash-canvas 의 dot grid 를 override. */
.dash-canvas.edit-mode {
background-color: var(--v5-bg);
background-image:
linear-gradient(0deg, rgba(var(--v5-primary-rgb), .035), rgba(var(--v5-primary-rgb), .035));
box-shadow: inset 0 0 0 2px rgba(var(--v5-primary-rgb), .22);
}
.dash-canvas.edit-mode::before {
content: '';
position: absolute; inset: 0;
background-image:
linear-gradient(to right, rgba(var(--v5-primary-rgb), .08) 1px, transparent 1px),
linear-gradient(to bottom, rgba(var(--v5-primary-rgb), .08) 1px, transparent 1px);
background-size: 20px 20px;
pointer-events: none;
z-index: 0;
animation: ud-grid-fade .4s var(--v5-ease-move) both;
}
.dash-canvas.control-mode {
background-color: var(--v5-bg);
background-image:
linear-gradient(0deg, rgba(var(--v5-cyan-rgb), .035), rgba(var(--v5-cyan-rgb), .035));
box-shadow: inset 0 0 0 2px rgba(var(--v5-cyan-rgb), .22);
overflow: auto;
}
.dash-canvas.control-mode::before {
content: '';
position: absolute; inset: 0;
background-image:
linear-gradient(to right, rgba(var(--v5-cyan-rgb), .08) 1px, transparent 1px),
linear-gradient(to bottom, rgba(var(--v5-cyan-rgb), .08) 1px, transparent 1px);
background-size: 24px 24px;
pointer-events: none;
z-index: 0;
animation: ud-grid-fade .4s var(--v5-ease-move) both;
}
/* 다크모드에서는 tint를 조금 강하게 */
.dark .dash-canvas.edit-mode {
background-image:
linear-gradient(0deg, rgba(var(--v5-primary-rgb), .05), rgba(var(--v5-primary-rgb), .05));
box-shadow: inset 0 0 0 2px rgba(var(--v5-primary-rgb), .28);
}
.dark .dash-canvas.control-mode {
background-image:
linear-gradient(0deg, rgba(var(--v5-cyan-rgb), .05), rgba(var(--v5-cyan-rgb), .05));
box-shadow: inset 0 0 0 2px rgba(var(--v5-cyan-rgb), .28);
}
@keyframes ud-grid-fade {
from { opacity: 0; }
to { opacity: 1; }
}
/* 카드 축소 + 클릭 가능 */
.dash-canvas.control-mode .dash-card {
transition: all .5s cubic-bezier(.16,1,.3,1);
opacity: .5; z-index: 25; cursor: pointer;
}
.dash-canvas.control-mode .dash-card:hover { opacity: .8; box-shadow: 0 0 20px var(--ctrl-cyan-glow); }
.dash-canvas.control-mode .dash-card.flow-active {
opacity: 1; border-color: var(--ctrl-cyan);
box-shadow: 0 0 30px rgba(var(--v5-cyan-rgb),.3);
}
/* ── 제어 모드 토글 버튼 활성 ── */
.dash-btn.control-on {
background: linear-gradient(135deg, var(--ctrl-cyan), #55efc4) !important;
color: #06050e !important; border-color: transparent !important;
box-shadow: 0 0 20px rgba(var(--v5-cyan-rgb),.3) !important; font-weight: 700;
}
.dash-btn.control-on:hover { box-shadow: 0 0 30px rgba(var(--v5-cyan-rgb),.45) !important; }
/* ═══ SVG 오버레이 ═══ */
.ctrl-svg {
position: absolute; inset: 0; width: 100%; height: 100%;
pointer-events: none; z-index: 10; overflow: visible;
}
/* ── 연결선 4종 ── */
.ctrl-line { fill: none; stroke: var(--ctrl-cyan); stroke-width: 1.5; opacity: .55;
stroke-dasharray: 6 3; animation: ctrlPulse 1.5s linear infinite; }
.ctrl-line-auto { fill: none; stroke: var(--ctrl-primary); stroke-width: 2.5; opacity: .6;
stroke-dasharray: 6 4; animation: ctrlPulse 1.2s linear infinite; }
.ctrl-line-cond { fill: none; stroke: var(--ctrl-amber); stroke-width: 2; opacity: .55;
stroke-dasharray: 4 4; animation: ctrlPulse 1.8s linear infinite; }
.ctrl-line-tpl { fill: none; stroke: var(--ctrl-pink); stroke-width: 2.5; opacity: .65;
stroke-dasharray: 5 5; animation: ctrlPulse 1.4s linear infinite; }
@keyframes ctrlPulse { to { stroke-dashoffset: -18; } }
/* ★ 라이트 모드 보정 */
html:not(.dark) .ctrl-line { stroke: #00a89e; stroke-width: 2; opacity: .5; }
html:not(.dark) .ctrl-line-auto { stroke: #5b4acf; stroke-width: 3; opacity: .6; }
html:not(.dark) .ctrl-line-cond { stroke: #d4a017; stroke-width: 2.5; opacity: .55; }
html:not(.dark) .ctrl-line-tpl { stroke: #e0559e; stroke-width: 3; opacity: .6; }
/* ═══ 연결선 위 뱃지 ═══ */
.ctrl-badge {
position: absolute; padding: .2rem .6rem; border-radius: 9px;
background: var(--v5-surface-solid);
border: 1px solid rgba(var(--v5-cyan-rgb),.3);
font-size: .55rem; font-weight: 700; color: var(--ctrl-cyan);
white-space: nowrap; z-index: 15; cursor: pointer;
transition: all .25s var(--v5-ease-move);
box-shadow: 0 4px 16px rgba(var(--v5-cyan-rgb),.12), 0 0 12px rgba(var(--v5-cyan-rgb),.1);
pointer-events: auto; transform: translate(-50%, -50%);
}
.ctrl-badge:hover {
border-color: var(--ctrl-cyan);
box-shadow: 0 8px 24px rgba(var(--v5-cyan-rgb),.3); transform: translate(-50%,-50%) scale(1.05);
}
.ctrl-badge.auto { border-color: rgba(var(--v5-primary-rgb),.35); color: var(--ctrl-primary);
box-shadow: 0 4px 16px rgba(var(--v5-primary-rgb),.12); }
.ctrl-badge.tpl-link { border-color: rgba(var(--v5-pink-rgb),.35); color: var(--ctrl-pink);
box-shadow: 0 4px 16px rgba(var(--v5-pink-rgb),.12); font-size: .5rem; }
/* 조건분기 뱃지 */
.ctrl-badge.cond {
border-color: rgba(253,203,110,.4); color: var(--v5-text, #e8e8ee);
box-shadow: 0 6px 20px rgba(253,203,110,.15); padding: .45rem .7rem;
min-width: 120px; text-align: left; white-space: normal; line-height: 1.4;
border-radius: 10px; border-width: 2px;
}
.ctrl-badge.cond .cb-head {
display: flex; align-items: center; gap: .3rem;
font-size: .5rem; font-weight: 700; color: var(--ctrl-amber); margin-bottom: .25rem;
}
.ctrl-badge.cond .cb-icon {
width: 16px; height: 16px; border-radius: 4px;
display: flex; align-items: center; justify-content: center; font-size: .6rem;
background: rgba(253,203,110,.15); border: 1px solid rgba(253,203,110,.3);
}
.ctrl-badge.cond .cb-cond {
font-size: .6rem; font-weight: 600; padding: .2rem .4rem;
border-radius: 6px; background: rgba(253,203,110,.08);
border: 1px dashed rgba(253,203,110,.25); margin-bottom: .25rem;
}
.ctrl-badge.cond .cb-paths { display: flex; gap: .5rem; font-size: .48rem; font-weight: 700; }
.ctrl-badge.cond .cb-yes { color: var(--ctrl-green); }
.ctrl-badge.cond .cb-yes::before { content: '●'; margin-right: .15rem; font-size: .4rem; }
.ctrl-badge.cond .cb-no { color: var(--v5-text-muted, #888); }
.ctrl-badge.cond .cb-no::before { content: '○'; margin-right: .15rem; font-size: .4rem; }
html:not(.dark) .ctrl-badge { background: rgba(255,255,255,.92);
border-color: rgba(0,0,0,.15); box-shadow: 0 2px 12px rgba(0,0,0,.1); }
/* ═══ 테이블 노드 ═══ */
.tbl-node {
position: absolute; width: 220px;
background: var(--v5-surface-solid);
border: 1px solid var(--ctrl-glass-border); border-radius: 12px;
box-shadow: 0 8px 24px rgba(0,0,0,.08), 0 0 20px rgba(var(--v5-cyan-rgb),.12);
z-index: 20; overflow: hidden; transition: border-color .2s var(--v5-ease-move), box-shadow .2s var(--v5-ease-move);
}
/* ═══════════════════════════════════════════════════════════════════════════
.tbl-node-compact — V3RuleNode 와 일관된 컴팩트 테이블 카드
(Phase 1: 컬럼별 port 폐기, 좌·우 edge 단일 port. 컬럼 선택은 NodeConfigPopover dropdown)
═══════════════════════════════════════════════════════════════════════════ */
.tbl-node-compact {
width: 180px;
border: 1.5px solid rgba(var(--v5-cyan-rgb), .4);
border-radius: 9px;
background: var(--v5-surface-solid);
box-shadow: 0 4px 12px -4px rgba(0,0,0,.1);
overflow: visible;
cursor: grab;
user-select: none;
transition: border-color .2s var(--v5-ease-move), box-shadow .2s var(--v5-ease-move);
}
.tbl-node-compact:hover {
border-color: rgba(var(--v5-cyan-rgb), .7);
box-shadow: 0 6px 18px -4px rgba(0,0,0,.14), 0 0 16px rgba(var(--v5-cyan-rgb), .15);
}
.tbl-node-compact:active { cursor: grabbing; }
.tbl-node-compact .tbl-node-stripe {
height: 3px;
background: var(--ctrl-cyan);
opacity: .9;
border-radius: 9px 9px 0 0;
}
.tbl-node-compact .tbl-node-head {
display: flex; align-items: center; gap: 6px;
padding: .45rem .6rem .35rem;
background: transparent;
border-bottom: none;
}
.tbl-node-compact .tbl-node-ico {
width: 20px; height: 20px; border-radius: 5px;
background: rgba(var(--v5-cyan-rgb), .14);
color: var(--ctrl-cyan);
display: grid; place-items: center;
flex-shrink: 0;
}
.tbl-node-compact .tbl-node-title { flex: 1; min-width: 0; display: flex; flex-direction: column; gap: 1px; }
.tbl-node-compact .tbl-node-label {
font-size: .73rem; font-weight: 700;
color: var(--v5-text); letter-spacing: -.01em;
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.tbl-node-compact .tbl-node-sub {
font-family: var(--v5-font-mono); font-size: .55rem;
color: var(--v5-text-muted); font-weight: 600;
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.tbl-node-compact .tbl-node-del {
width: 16px; height: 16px;
background: transparent; border: none; cursor: pointer;
display: grid; place-items: center;
color: var(--v5-text-muted);
border-radius: 4px;
flex-shrink: 0;
}
.tbl-node-compact .tbl-node-del:hover {
background: rgba(255, 71, 87, .15);
color: rgb(255, 71, 87);
}
.tbl-node-compact .tbl-node-stats {
padding: .3rem .6rem .45rem;
margin-top: 2px;
display: flex; gap: 5px; justify-content: flex-start;
font-family: var(--v5-font-mono);
font-size: .5rem; color: var(--v5-text-muted);
border-top: 1px dashed var(--v5-border);
}
/* 컴팩트 모드 port — V3RuleNode 와 동일한 위치/사이즈 */
.tbl-node-compact > .ctrl-io-port.port-in {
width: 12px; height: 12px;
left: -6px; top: 50%; transform: translateY(-50%);
border: 2px solid var(--ctrl-cyan);
background: var(--v5-surface-solid);
border-radius: 50%;
z-index: 25;
}
.tbl-node-compact > .ctrl-an-ports-out {
position: absolute;
right: -6px; top: 50%; transform: translateY(-50%);
display: flex; flex-direction: column; gap: 8px;
}
.tbl-node-compact > .ctrl-an-ports-out .ctrl-io-port {
position: relative; right: auto; top: auto; transform: none;
width: 12px; height: 12px;
background: var(--ctrl-cyan); border: none; border-radius: 50%;
}
.dark .tbl-node { box-shadow: 0 8px 24px rgba(0,0,0,.5), 0 0 20px rgba(var(--v5-cyan-rgb),.06); }
.tbl-node:hover {
border-color: var(--ctrl-cyan);
box-shadow: 0 12px 32px rgba(0,0,0,.12), 0 0 30px rgba(var(--v5-cyan-rgb),.18);
}
.tbl-node-head {
display: flex; align-items: center; gap: .4rem; padding: .55rem .7rem;
background: linear-gradient(135deg, rgba(var(--v5-cyan-rgb),.12), rgba(var(--v5-cyan-rgb),.04));
border-bottom: 1px solid rgba(var(--v5-cyan-rgb),.15); cursor: grab;
}
.tbl-node-head:active { cursor: grabbing; }
.tbl-icon {
width: 20px; height: 20px; border-radius: 5px;
display: flex; align-items: center; justify-content: center;
font-size: .7rem; background: rgba(var(--v5-cyan-rgb),.12); flex-shrink: 0;
}
.tbl-name { flex: 1; font-size: .68rem; font-weight: 700; color: var(--v5-text, #e8e8ee); letter-spacing: -.01em; }
.tbl-badge {
font-family: var(--v5-font-mono);
font-size: .5rem; padding: .12rem .4rem; border-radius: 999px;
background: var(--v5-bg-subtle, rgba(255,255,255,.04));
color: var(--v5-text-muted, #888);
font-weight: 600; letter-spacing: -.01em;
border: 1px solid var(--v5-border, rgba(255,255,255,.08));
}
.tbl-node-cols { padding: .35rem 0; max-height: 220px; overflow-y: auto; overflow-x: hidden; }
.tbl-col {
position: relative;
display: flex; align-items: center; gap: .35rem;
padding: .28rem .9rem; /* 좌·우 균등 — 3px stripe + 여백 */
font-size: .6rem; color: var(--v5-text-sec, #aaa); transition: background .1s;
}
.tbl-col:hover { background: var(--v5-surface-hover, rgba(255,255,255,.05)); color: var(--v5-text, #eee); }
.tbl-port {
width: 8px; height: 8px; border-radius: 50%; background: var(--v5-surface, #2a2a36);
border: 2px solid rgba(var(--v5-cyan-rgb),.35); flex-shrink: 0; cursor: crosshair; transition: all .2s;
}
.tbl-port:hover {
background: var(--ctrl-cyan); border-color: var(--ctrl-cyan);
box-shadow: 0 0 8px rgba(var(--v5-cyan-rgb),.5); transform: scale(1.3);
}
.tbl-port.pk { border-color: var(--ctrl-primary); background: rgba(var(--v5-primary-rgb),.15); }
.tbl-port.fk { border-color: var(--ctrl-amber); background: rgba(253,203,110,.15); }
.tbl-col-name { flex: 1; font-weight: 500; }
.tbl-col-type { font-size: .45rem; color: var(--v5-text-muted, #777); font-weight: 600; margin-left: auto; }
.tbl-col-mark { font-size: .42rem; font-weight: 700; padding: .05rem .25rem; border-radius: 4px; margin-left: .2rem; }
.tbl-col-mark.pk { color: var(--ctrl-primary); background: rgba(var(--v5-primary-rgb),.1); }
.tbl-col-mark.fk { color: var(--ctrl-amber); background: rgba(253,203,110,.1); }
/* ═══ 제어 노드 (액션/조건/타이머) ═══ */
.ctrl-action-node {
position: absolute; width: 160px;
background: var(--v5-surface-solid);
border: 1px solid rgba(var(--na-rgb, 0,206,201), .25); border-radius: 12px;
box-shadow: 0 8px 24px rgba(0,0,0,.08), 0 0 16px rgba(var(--na-rgb, 0,206,201), .12);
z-index: 20; overflow: visible; transition: border-color .2s var(--v5-ease-move), box-shadow .2s var(--v5-ease-move);
}
.ctrl-action-node:hover {
border-color: rgba(var(--na-rgb, 0,206,201), .5);
box-shadow: 0 12px 32px rgba(0,0,0,.12), 0 0 24px rgba(var(--na-rgb, 0,206,201), .18);
}
.ctrl-an-head {
display: flex; align-items: center; gap: .4rem; padding: .5rem .65rem;
background: linear-gradient(135deg, rgba(var(--na-rgb, 0,206,201), .1), rgba(var(--na-rgb, 0,206,201), .03));
border-bottom: 1px solid rgba(var(--na-rgb, 0,206,201), .12);
border-radius: 12px 12px 0 0; cursor: grab;
}
.ctrl-an-head:active { cursor: grabbing; }
.ctrl-an-icon {
width: 22px; height: 22px; border-radius: 6px;
display: flex; align-items: center; justify-content: center; font-size: .8rem;
background: linear-gradient(135deg, rgba(var(--na-rgb, 0,206,201), .18), rgba(var(--na-rgb, 0,206,201), .06));
border: 1px solid rgba(var(--na-rgb, 0,206,201), .25);
}
.ctrl-an-name { flex: 1; font-size: .62rem; font-weight: 700; color: var(--v5-text, #e8e8ee); }
.ctrl-an-del {
width: 18px; height: 18px; border-radius: 5px;
border: 1px solid transparent; background: transparent;
color: var(--v5-text-muted, #888); font-size: .55rem; cursor: pointer;
display: flex; align-items: center; justify-content: center; transition: all .15s;
}
.ctrl-an-del:hover {
background: rgba(var(--v5-red-rgb),.1); border-color: rgba(var(--v5-red-rgb),.3); color: var(--ctrl-red);
}
.ctrl-an-body {
padding: .5rem .65rem; cursor: pointer;
border-radius: 0 0 12px 12px; transition: background .15s;
}
.ctrl-an-body:hover { background: rgba(var(--na-rgb, 0,206,201), .06); }
.ctrl-an-summary { font-size: .55rem; color: var(--v5-text-sec, #aaa); line-height: 1.4; }
/* ═══ I/O 포트 ═══ */
.ctrl-io-port {
position: absolute; width: 12px; height: 12px; border-radius: 50%;
border: 2px solid; cursor: crosshair; transition: all .2s; z-index: 25;
}
/* 모든 port 의 hit area 확장 — ::before 로 24x24 invisible zone */
.ctrl-io-port::before {
content: ""; position: absolute; inset: -6px;
border-radius: 50%;
}
.ctrl-io-port.port-in {
left: -6px; top: 50%; transform: translateY(-50%);
border-color: var(--ctrl-cyan); background: var(--v5-surface, #2a2a36);
}
/* 테이블 컬럼 port — 짧은 수직 stripe (node cat-stripe 와 일관) */
.ctrl-io-port.tbl-io {
width: 3px; height: 60%;
border: none; border-radius: 2px;
background: var(--ctrl-cyan); opacity: .7;
cursor: crosshair;
transition: opacity .12s linear; /* width 트랜지션 제거 (밀려나는 느낌 없애기) */
}
.ctrl-io-port.port-in.tbl-io {
left: 0; right: auto; top: 50%; transform: translateY(-50%);
}
.ctrl-io-port.port-out.tbl-io {
position: absolute; left: auto; right: 0; top: 50%; transform: translateY(-50%);
}
/* PK 컬럼 stripe — 보라 강조 */
.ctrl-io-port.port-in.tbl-io.pk,
.ctrl-io-port.port-out.tbl-io.pk { background: rgb(var(--v5-primary-rgb)); }
/* hover/active — opacity + glow 만 (width 변화 X) */
.ctrl-io-port.tbl-io:hover {
opacity: 1;
box-shadow: 0 0 8px rgba(var(--v5-cyan-rgb), .55);
}
.ctrl-io-port.port-hover.tbl-io {
opacity: 1 !important;
box-shadow: 0 0 12px rgba(var(--v5-cyan-rgb), .85) !important;
}
.ctrl-io-port.port-out { border-color: var(--ctrl-cyan); background: var(--ctrl-cyan); }
.ctrl-io-port.port-yes { border-color: var(--ctrl-green); background: var(--ctrl-green); }
.ctrl-io-port.port-no { border-color: var(--v5-text-muted, #888); background: var(--v5-text-muted, #888); opacity: .6; }
.ctrl-an-ports-out {
position: absolute; right: -6px; top: 50%; transform: translateY(-50%);
display: flex; flex-direction: column; gap: 8px;
}
.ctrl-an-ports-out .ctrl-io-port { position: relative; right: auto; top: auto; transform: none; }
.port-label {
position: absolute; right: 14px; top: 50%; transform: translateY(-50%);
font-size: .42rem; font-weight: 700; color: var(--v5-text-muted, #888);
pointer-events: none; white-space: nowrap;
}
.ctrl-io-port:hover { box-shadow: 0 0 10px rgba(var(--v5-cyan-rgb),.5); transform: scale(1.3); }
.ctrl-io-port.port-in:hover { background: var(--ctrl-cyan); transform: translateY(-50%) scale(1.3); }
.ctrl-io-port.port-in.tbl-io:hover { transform: scale(1.3); }
.ctrl-io-port.port-hover {
background: var(--ctrl-cyan) !important;
box-shadow: 0 0 14px rgba(var(--v5-cyan-rgb),.6) !important;
transform: translateY(-50%) scale(1.5) !important;
}
.ctrl-io-port.port-hover.tbl-io { transform: scale(1.5) !important; }
/* 드래그 중 모든 input 포트 pulse */
.dash-canvas.port-dragging .ctrl-io-port.port-in { animation: portPulse 1.2s ease infinite; }
@keyframes portPulse {
0%, 100% { box-shadow: 0 0 4px rgba(var(--v5-cyan-rgb),.3); }
50% { box-shadow: 0 0 14px rgba(var(--v5-cyan-rgb),.6); }
}
/* ═══ 규칙 연결선 — mockup v3 EditCanvas: solid + 둥근 join, no port 만 dashed ═══ */
.rule-temp-line {
fill: none; stroke: var(--ctrl-cyan); stroke-width: 2;
stroke-dasharray: 6 3; /* 드래그 중 임시선만 점선 (cursor 따라가는 중임을 강조) */
opacity: .7; pointer-events: none;
}
.rule-conn-path {
fill: none; stroke: var(--ctrl-cyan); stroke-width: 2.2; opacity: .75;
stroke-linecap: round; stroke-linejoin: round;
}
.rule-conn-path.conn-yes { stroke: var(--ctrl-green); }
.rule-conn-path.conn-no { stroke: var(--v5-text-muted, #888); opacity: .55; stroke-dasharray: 5 4; }
/* Phase 3: edge_type 별 색상 — yes/no 가 우선이라 더 낮은 specificity */
.rule-conn-path.edge-data-context { stroke: var(--ctrl-cyan); } /* 테이블 → 노드 (데이터 입력) */
.rule-conn-path.edge-execution-flow { stroke: var(--ctrl-primary); } /* 노드 → 노드 (실행 순서) */
.rule-conn-path.edge-table-mutation { stroke: rgb(253, 121, 168); } /* 노드 → 테이블 (저장/수정) */
.rule-conn-path.edge-lookup { stroke: var(--ctrl-green); opacity: .65; stroke-dasharray: 6 4; } /* 테이블 → 테이블 (FK) */
/* yes/no override 우선 — 명시 specificity */
.rule-conn-path.conn-yes.edge-data-context,
.rule-conn-path.conn-yes.edge-execution-flow,
.rule-conn-path.conn-yes.edge-table-mutation,
.rule-conn-path.conn-yes.edge-lookup { stroke: var(--ctrl-green); }
.rule-conn-path.conn-no.edge-data-context,
.rule-conn-path.conn-no.edge-execution-flow,
.rule-conn-path.conn-no.edge-table-mutation,
.rule-conn-path.conn-no.edge-lookup { stroke: var(--v5-text-muted, #888); stroke-dasharray: 5 4; opacity: .55; }
/* 연결 삭제 버튼 */
.rule-conn-badge {
position: absolute; transform: translate(-50%, -50%);
z-index: 15; pointer-events: auto; padding: 6px;
}
.conn-x {
display: flex; align-items: center; justify-content: center;
width: 20px; height: 20px; border-radius: 50%;
background: var(--v5-surface-solid);
border: 1.5px solid rgba(var(--v5-red-rgb),.15);
color: var(--v5-text-muted, #888); font-size: .55rem; font-weight: 700; cursor: pointer;
opacity: 0; transition: all .2s var(--v5-ease-move); transform: scale(.6);
}
.rule-conn-badge:hover .conn-x {
opacity: 1; transform: scale(1);
background: rgba(var(--v5-red-rgb),.15); border-color: rgba(var(--v5-red-rgb),.5); color: var(--ctrl-red);
box-shadow: 0 2px 12px rgba(var(--v5-red-rgb),.2);
}
/* ═══ 설정 팝오버 ═══ */
.ctrl-cfg-pop {
position: absolute; width: 220px;
background: var(--v5-surface-solid);
border: 1px solid rgba(var(--v5-primary-rgb),.3); border-radius: 12px;
box-shadow: 0 12px 40px rgba(0,0,0,.15), 0 0 24px rgba(var(--v5-primary-rgb),.18);
z-index: 50; padding: .7rem;
opacity: 0; transform: translateX(-8px);
transition: opacity .2s var(--v5-ease-move), transform .2s var(--v5-ease-move);
}
.ctrl-cfg-pop.open { opacity: 1; transform: translateX(0); }
.cfg-hd { font-size: .65rem; font-weight: 700; color: var(--v5-text, #eee);
padding-bottom: .5rem; border-bottom: 1px solid var(--v5-border, #3a3a48); margin-bottom: .5rem; }
.cfg-sec { margin-bottom: .5rem; }
.cfg-lb { display: block; font-size: .48rem; font-weight: 700; color: var(--v5-text-muted, #888);
text-transform: uppercase; letter-spacing: .05em; margin-bottom: .15rem; }
.cfg-sel, .cfg-inp, .cfg-ta {
width: 100%; padding: .3rem .45rem; border-radius: 7px;
border: 1px solid var(--v5-border, #3a3a48); background: var(--v5-surface, #2a2a36);
color: var(--v5-text, #eee); font-size: .55rem; font-family: inherit;
outline: none; transition: border-color .15s; box-sizing: border-box;
}
.cfg-sel:focus, .cfg-inp:focus, .cfg-ta:focus { border-color: var(--ctrl-primary); }
.cfg-add-btn {
width: 100%; padding: .25rem; border-radius: 6px;
border: 1px dashed var(--v5-border, #3a3a48); background: transparent;
color: var(--v5-text-muted, #888); font-size: .5rem; cursor: pointer; margin-top: .3rem;
}
.cfg-add-btn:hover { border-color: var(--ctrl-cyan); color: var(--ctrl-cyan); }
.cfg-ft {
display: flex; gap: .3rem; padding-top: .5rem;
border-top: 1px solid var(--v5-border, #3a3a48); margin-top: .5rem;
}
.cfg-btn {
flex: 1; padding: .3rem; border-radius: 7px;
border: 1px solid var(--v5-border, #3a3a48); background: var(--v5-surface, #2a2a36);
color: var(--v5-text, #eee); font-size: .55rem; font-weight: 600;
cursor: pointer; transition: all .15s;
}
.cfg-btn.save { background: var(--ctrl-primary); border-color: var(--ctrl-primary); color: #fff; }
.cfg-btn:hover { opacity: .85; }
/* Phase 2: schema-driven dropdown 보조 클래스 */
.cfg-empty {
padding: .45rem .55rem; border-radius: 7px;
background: rgba(255, 165, 2, .08);
border: 1px dashed rgba(255, 165, 2, .35);
color: var(--v5-text-muted, #888);
font-size: .52rem; line-height: 1.5;
}
.cfg-static {
display: flex; align-items: center; gap: 5px;
padding: .3rem .45rem; border-radius: 7px;
background: rgba(var(--v5-cyan-rgb), .06);
border: 1px solid rgba(var(--v5-cyan-rgb), .25);
font-size: .55rem;
}
.cfg-static-main { color: var(--v5-text); font-weight: 600; }
.cfg-static-sub {
font-family: var(--v5-font-mono); font-size: .5rem;
color: var(--v5-text-muted);
}
.cfg-static-hint {
margin-left: auto;
font-family: var(--v5-font-mono); font-size: .45rem;
color: var(--ctrl-cyan); opacity: .7;
}
/* native select 의 optgroup label 색감 정리 */
.cfg-sel optgroup {
font-family: var(--v5-font-mono); font-size: .5rem;
color: var(--ctrl-cyan); font-style: normal; font-weight: 700;
}
.cfg-sel option { color: var(--v5-text); font-size: .55rem; }
/* ═══ 팔레트 ═══ */
.ctrl-palette-section {
font-size: .52rem; font-weight: 700; color: var(--ctrl-cyan);
text-transform: uppercase; letter-spacing: .08em; padding: .7rem .65rem .3rem;
}
.ctrl-palette-item {
display: flex; align-items: center; gap: .5rem; padding: .45rem .65rem;
border-radius: 8px; font-size: .68rem; font-weight: 500;
color: var(--v5-text-sec, #aaa); cursor: grab; transition: all .2s;
}
.ctrl-palette-item:hover {
background: rgba(var(--v5-cyan-rgb),.08); color: var(--v5-text, #eee); transform: translateX(2px);
}
.ctrl-palette-item .cp-icon { font-size: .8rem; width: 20px; text-align: center; }
.ctrl-palette-item[draggable="true"] { cursor: grab; }
.ctrl-palette-item[draggable="true"]:active { cursor: grabbing; }
/* ═══ 제어 모드 툴바 ═══ */
.ctrl-toolbar {
display: flex; align-items: center; gap: .5rem; padding: .3rem .5rem;
background: var(--ctrl-glass); border-bottom: 1px solid var(--ctrl-glass-border);
font-size: .6rem;
}
.ctrl-toolbar-mode {
display: flex; gap: .25rem;
}
.ctrl-mode-btn {
padding: .25rem .6rem; border-radius: 6px; border: 1px solid var(--v5-border, #3a3a48);
background: transparent; color: var(--v5-text-sec, #aaa); font-size: .55rem;
font-weight: 600; cursor: pointer; transition: all .2s;
}
.ctrl-mode-btn.on {
background: rgba(var(--v5-cyan-rgb),.12); border-color: var(--ctrl-cyan);
color: var(--ctrl-cyan); font-weight: 700;
}
.ctrl-mode-btn:hover:not(.on) { background: var(--v5-surface-hover, rgba(255,255,255,.04)); }
html:not(.dark) .ctrl-action-node {
box-shadow: 0 4px 16px rgba(0,0,0,.06), 0 0 12px rgba(var(--na-rgb, 0,206,201), .06);
}
html:not(.dark) .rule-conn-path { stroke-width: 2.5; opacity: .5; }
html:not(.dark) .ctrl-cfg-pop {
box-shadow: 0 8px 32px rgba(0,0,0,.1); border-color: rgba(var(--v5-primary-rgb),.2);
}
/* ═══════════════════════════════════════════════════════════════════════
카드 단위 제어 모드 (2026-05-19 v4) — mockup 컨셉
- 카드 클릭 → 그 카드 좌측 상단으로 축소(강제 inline 덮어쓰기) + 강조
- 그 카드 우측 옆에 부속 CardPanel (모드 토글 + 액션 + ✕)
- 우측 빈 공간에 트리 자람(view) 또는 룰 빌더(edit)
═══════════════════════════════════════════════════════════════════════ */
.ctrl-mode-root {
position: absolute; inset: 0; z-index: 30;
pointer-events: none;
}
.ctrl-mode-root > * { pointer-events: auto; }
/* 좌상단 안내 (카드 미선택 시) */
.ctrl-mode-hint {
position: absolute; top: 14px; left: 14px;
display: inline-flex; align-items: center; gap: .4rem;
padding: .4rem .75rem; border-radius: 999px;
background: var(--v5-surface-solid);
border: 1.5px solid var(--ctrl-cyan);
color: var(--ctrl-cyan);
font-size: .6rem; font-weight: 700;
box-shadow: var(--v5-glow-cyan-sm);
z-index: 35;
animation: ctrlHintPulse 2.4s ease-in-out infinite;
}
@keyframes ctrlHintPulse {
0%, 100% { box-shadow: var(--v5-glow-cyan-sm); }
50% { box-shadow: 0 0 24px rgba(var(--v5-cyan-rgb), .5); }
}
/* ─────────────────────────────────────────────────────────────────────
카드 fade — 제어 모드 진입 시 자동
───────────────────────────────────────────────────────────────────── */
/* 기본: 약한 fade + 클릭 가능 (카드 미선택 상태) */
.dash-canvas.control-mode [data-card-id] {
cursor: pointer;
opacity: .55;
transition: all .35s cubic-bezier(.16, 1, .3, 1);
}
.dash-canvas.control-mode [data-card-id]:hover {
opacity: .92;
box-shadow: var(--v5-glow-cyan-sm), 0 0 28px rgba(var(--v5-cyan-rgb), .35);
}
.dash-canvas.control-mode [data-card-id] > * {
pointer-events: none;
}
/* 선택된 카드 = 좌측 상단 강제 축소 (inline style !important 로 덮어쓰기) */
.dash-canvas.control-mode [data-card-id][data-flow-active="1"] {
left: 20px !important;
top: 80px !important;
width: 320px !important;
height: 220px !important;
opacity: 1 !important;
z-index: 25;
box-shadow: 0 0 36px rgba(var(--v5-cyan-rgb), .55), var(--v5-glow-md) !important;
border: 1.5px solid var(--ctrl-cyan);
cursor: default;
overflow: hidden;
}
.dash-canvas.control-mode [data-card-id][data-flow-active="1"]:hover {
opacity: 1 !important;
}
.dash-canvas.control-mode [data-card-id][data-flow-active="1"] > * {
pointer-events: none;
opacity: .85;
}
/* 선택된 카드 외 — 더 강하게 fade (구분) */
.dash-canvas.control-mode:has([data-flow-active="1"]) [data-card-id]:not([data-flow-active="1"]) {
opacity: .12;
pointer-events: none;
cursor: default;
}
.dash-canvas.control-mode:has([data-flow-active="1"]) [data-card-id]:not([data-flow-active="1"]):hover {
opacity: .12;
box-shadow: none;
}
/* edit 모드 — 선택 카드도 좀 더 흐리게 (룰 빌더가 더 전면) */
.dash-canvas.control-mode-edit [data-card-id][data-flow-active="1"] {
opacity: .55 !important;
}
.dash-canvas.control-mode-edit [data-card-id]:not([data-flow-active="1"]) {
opacity: .06 !important;
}
/* ─────────────────────────────────────────────────────────────────────
ControlCardPanel — 선택 카드 우측 옆 floating
카드는 left:20, w:320 → 패널은 left:360 부근에 떠 있음
───────────────────────────────────────────────────────────────────── */
.ctrl-card-panel {
position: absolute; left: 360px; top: 80px;
width: 250px;
display: flex; flex-direction: column; gap: .45rem;
padding: .6rem .7rem;
background: var(--v5-surface-solid);
border: 1.5px solid var(--ctrl-cyan);
border-radius: 12px;
box-shadow: 0 12px 32px rgba(0, 0, 0, .15), var(--v5-glow-cyan-sm);
z-index: 40;
animation: ctrlPanelSlide .35s cubic-bezier(.16, 1, .3, 1);
}
@keyframes ctrlPanelSlide {
from { opacity: 0; transform: translateX(-12px); }
to { opacity: 1; transform: translateX(0); }
}
.ctrl-card-panel-head {
display: flex; align-items: center; gap: .4rem;
padding-bottom: .35rem;
border-bottom: 1px solid var(--v5-border-subtle);
}
.ctrl-card-panel-icon {
width: 24px; height: 24px; border-radius: 7px;
display: flex; align-items: center; justify-content: center;
background: linear-gradient(135deg, rgba(var(--v5-cyan-rgb), .25), rgba(var(--v5-cyan-rgb), .08));
border: 1px solid rgba(var(--v5-cyan-rgb), .3);
font-size: .8rem;
flex-shrink: 0;
}
.ctrl-card-panel-title-wrap {
flex: 1; min-width: 0;
}
.ctrl-card-panel-title {
font-size: .65rem; font-weight: 700; color: var(--v5-text);
overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
}
.ctrl-card-panel-type {
font-size: .42rem; font-weight: 600; color: var(--v5-text-muted);
text-transform: uppercase; letter-spacing: .04em; margin-top: 1px;
}
.ctrl-card-panel-close {
width: 20px; height: 20px; border-radius: 5px;
display: flex; align-items: center; justify-content: center;
background: transparent; border: 1px solid var(--v5-border);
color: var(--v5-text-muted); cursor: pointer;
transition: all .15s;
flex-shrink: 0;
}
.ctrl-card-panel-close:hover {
border-color: var(--ctrl-red);
color: var(--ctrl-red);
background: rgba(255, 71, 87, .08);
}
.ctrl-card-panel-source { display: flex; }
.ctrl-card-panel-source-chip {
display: inline-flex; align-items: center; gap: .25rem;
padding: .15rem .45rem; border-radius: 999px;
background: rgba(var(--v5-pink-rgb), .1);
border: 1px solid rgba(var(--v5-pink-rgb), .35);
color: var(--ctrl-pink);
font-size: .5rem; font-weight: 700; font-family: monospace;
user-select: none;
max-width: 100%;
overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
}
.ctrl-card-panel-source-chip[draggable="true"] {
cursor: grab; transition: all .15s;
}
.ctrl-card-panel-source-chip[draggable="true"]:hover {
background: rgba(var(--v5-pink-rgb), .18);
border-color: var(--ctrl-pink);
box-shadow: 0 0 12px rgba(var(--v5-pink-rgb), .3);
}
.ctrl-card-panel-source-chip[draggable="true"]:active { cursor: grabbing; }
/* segmented mode toggle (카드 패널 안) */
.ctrl-card-panel-mode {
display: flex; gap: 2px;
padding: 2px; border-radius: 7px;
background: var(--v5-surface-hover);
border: 1px solid var(--v5-border-subtle);
}
.ctrl-card-panel-mode-btn {
flex: 1;
display: inline-flex; align-items: center; justify-content: center;
gap: .25rem;
padding: .3rem .4rem; border-radius: 5px;
border: none; background: transparent;
color: var(--v5-text-sec);
font-size: .55rem; font-weight: 600; cursor: pointer;
transition: all .15s;
}
.ctrl-card-panel-mode-btn:hover:not(.on) {
color: var(--v5-text);
background: rgba(var(--v5-cyan-rgb), .06);
}
.ctrl-card-panel-mode-btn.on {
background: linear-gradient(135deg, var(--ctrl-cyan), #2ed5d0);
color: #0a0a0f; font-weight: 700;
box-shadow: 0 2px 6px rgba(var(--v5-cyan-rgb), .35);
}
.ctrl-card-panel-actions {
display: flex; gap: .3rem;
}
.ctrl-card-panel-btn {
display: inline-flex; align-items: center; justify-content: center;
gap: .2rem;
padding: .3rem .45rem; border-radius: 6px;
border: 1px solid var(--v5-border);
background: var(--v5-surface-solid);
color: var(--v5-text-sec);
font-size: .5rem; font-weight: 600; cursor: pointer;
transition: all .15s;
flex: 1;
overflow: hidden;
}
.ctrl-card-panel-btn:hover:not(:disabled) {
border-color: var(--ctrl-cyan);
color: var(--ctrl-cyan);
box-shadow: var(--v5-glow-cyan-sm);
}
.ctrl-card-panel-btn:disabled { opacity: .4; cursor: not-allowed; }
.ctrl-card-panel-btn.primary {
background: linear-gradient(135deg, var(--v5-primary), #8b7ee8);
border-color: var(--v5-primary);
color: #fff;
box-shadow: var(--v5-glow-sm);
}
.ctrl-card-panel-btn.primary:hover:not(:disabled) {
box-shadow: var(--v5-glow-md);
filter: brightness(1.08);
}
.ctrl-card-panel-status {
font-size: .45rem; font-weight: 600;
color: var(--v5-text-muted); text-align: center;
letter-spacing: .04em;
}
.ctrl-card-panel-hint {
font-size: .48rem; color: var(--v5-text-muted);
text-align: center; font-style: italic;
padding: .25rem 0;
}
.ctrl-card-panel-dropdown {
position: absolute; bottom: 100%; left: 0; margin-bottom: 4px;
min-width: 200px; max-height: 200px; overflow-y: auto;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 10px; padding: .25rem;
box-shadow: 0 -12px 32px rgba(0, 0, 0, .15), var(--v5-glow-sm);
z-index: 100;
}
.ctrl-card-panel-dropdown-item {
display: block; width: 100%; text-align: left;
padding: .3rem .5rem; border-radius: 5px;
border: none; background: transparent;
color: var(--v5-text-sec);
font-size: .52rem; font-weight: 500; cursor: pointer;
transition: all .12s;
}
.ctrl-card-panel-dropdown-item:hover {
background: rgba(var(--v5-cyan-rgb), .08);
color: var(--ctrl-cyan);
}
.ctrl-card-panel-dropdown-item.active {
background: rgba(var(--v5-cyan-rgb), .14);
color: var(--ctrl-cyan);
font-weight: 700;
}
/* ─────────────────────────────────────────────────────────────────────
❸ 팔레트 보강 (검색 + 추천 + max-height + 한국어)
───────────────────────────────────────────────────────────────────── */
.ctrl-palette {
display: flex; flex-direction: column;
flex: 1; min-height: 0; height: 100%;
overflow: hidden;
}
.ctrl-palette-header {
flex-shrink: 0;
display: flex; align-items: center; justify-content: space-between;
padding: .55rem .7rem .3rem;
border-bottom: 1px solid var(--v5-border-subtle);
}
.ctrl-palette-header-title {
font-size: .58rem; font-weight: 700; color: var(--ctrl-cyan);
text-transform: uppercase; letter-spacing: .08em;
}
.ctrl-palette-header-hint {
font-size: .48rem; color: var(--v5-text-muted); font-weight: 500;
}
.ctrl-palette-search-wrap {
flex-shrink: 0; position: relative;
padding: .4rem .65rem .15rem;
}
.ctrl-palette-search-icon {
position: absolute; left: 1rem; top: 50%; transform: translateY(-30%);
color: var(--v5-text-muted); pointer-events: none; z-index: 1;
}
.ctrl-palette-search {
width: 100%; padding: .3rem .5rem .3rem 1.45rem;
border-radius: 7px;
border: 1px solid var(--v5-border);
background: var(--v5-surface-solid);
color: var(--v5-text);
font-size: .55rem; font-family: inherit; outline: none;
transition: border-color .15s, box-shadow .15s;
}
.ctrl-palette-search::placeholder { color: var(--v5-text-muted); }
.ctrl-palette-search:focus {
border-color: var(--ctrl-cyan);
box-shadow: var(--v5-glow-cyan-sm);
}
.ctrl-palette-search:disabled { opacity: .4; cursor: not-allowed; }
.ctrl-palette-scroll {
flex: 1; min-height: 0; overflow-y: auto;
transition: opacity .15s;
}
.ctrl-palette-scroll.disabled { opacity: .35; }
.ctrl-palette-section-rec {
display: flex; align-items: center; gap: .2rem;
color: var(--v5-amber, #fdcb6e) !important;
}
.ctrl-palette-section-count {
margin-left: auto;
font-size: .42rem; font-weight: 600;
padding: .05rem .3rem; border-radius: 999px;
background: var(--v5-surface-hover);
color: var(--v5-text-muted);
letter-spacing: 0; text-transform: none;
}
.ctrl-palette-tables {
padding: 0 .2rem .3rem;
}
.ctrl-palette-tables-others {
max-height: 260px; overflow-y: auto;
border-bottom: 1px solid var(--v5-border-subtle);
margin-bottom: .25rem;
}
.ctrl-palette-item-rec {
background: rgba(var(--v5-amber-rgb, 253, 203, 110), .04);
position: relative;
}
.ctrl-palette-item-rec:hover {
background: rgba(var(--v5-amber-rgb, 253, 203, 110), .12);
}
.cp-rec-star {
color: var(--v5-amber, #fdcb6e);
fill: currentColor;
flex-shrink: 0;
}
.ctrl-palette-item .cp-label {
display: flex; flex-direction: column; gap: 1px;
min-width: 0; flex: 1; overflow: hidden;
}
.ctrl-palette-item .cp-label-main {
font-size: .62rem; font-weight: 600; color: var(--v5-text);
overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
line-height: 1.2;
}
.ctrl-palette-item .cp-label-sub {
font-size: .44rem; color: var(--v5-text-muted);
font-family: monospace;
overflow: hidden; text-overflow: ellipsis; white-space: nowrap;
line-height: 1.2;
}
.ctrl-palette-empty {
padding: .8rem .7rem; font-size: .55rem;
color: var(--v5-text-muted);
text-align: center; font-style: italic;
}
/* ── 라이트 모드 보정 ── */
html:not(.dark) .ctrl-overlay-backdrop { opacity: .88; }
html:not(.dark) .ctrl-card-hotspot {
background: rgba(var(--v5-cyan-rgb), .03);
border-color: rgba(var(--v5-cyan-rgb), .55);
}
html:not(.dark) .ctrl-card-hotspot:hover {
background: rgba(var(--v5-cyan-rgb), .06);
box-shadow: 0 0 24px rgba(var(--v5-cyan-rgb), .28), 0 8px 32px rgba(0, 0, 0, .08);
}
/* ═══════════════════════════════════════════════════════════════════════════
v3 시안 통합 — 2026-05-19
참조: INVYONE Design System (2).zip / control-mode/styles.css
/Users/gbpark/Downloads/control-mode.standalone.html
"Control IDE" takeover 디자인 — v5 토큰 100%, 즉흥 hex 금지
═══════════════════════════════════════════════════════════════════════════ */
/* ─── 제어 토글 — cyan + green 그라데이션 + bounce in ───
AppLayout 의 <button className="ud-htool on" data-mode="ctrl"> 적용 */
.ud-htool[data-mode="ctrl"].on {
background: linear-gradient(135deg,
rgba(var(--v5-cyan-rgb), .95),
rgba(var(--v5-green-rgb), .95)) !important;
color: #06050e !important;
border-color: transparent !important;
box-shadow:
0 0 0 1px rgba(var(--v5-cyan-rgb), .4),
0 0 40px rgba(var(--v5-cyan-rgb), .22) !important;
font-weight: 700;
animation: ctrl-toggle-in .45s var(--v5-ease-enter) both;
}
.ud-htool[data-mode="ctrl"].on:hover {
box-shadow:
0 0 0 1px rgba(var(--v5-cyan-rgb), .5),
0 0 50px rgba(var(--v5-cyan-rgb), .32) !important;
}
/* 그라데이션이 강조 역할을 하므로 기본 underline 은 숨김 */
.ud-htool[data-mode="ctrl"].on::before { display: none; }
@keyframes ctrl-toggle-in {
0% { transform: scale(.85); opacity: .4; }
60% { transform: scale(1.06); }
100% { transform: scale(1); opacity: 1; }
}
/* ─── 카테고리 칩 — palette/inspector/canvas 공용 (v3 .cat-chip + .c-*) ─── */
.cat-chip {
display: inline-flex; align-items: center; gap: 4px;
font-family: var(--v5-font-mono);
font-size: .55rem;
font-weight: 700;
letter-spacing: .05em;
text-transform: uppercase;
padding: 2px 6px;
border-radius: 999px;
border: 1px solid currentColor;
background: color-mix(in srgb, currentColor 10%, transparent);
}
.c-trigger { color: rgb(var(--v5-cyan-rgb)); }
.c-cond { color: rgb(var(--v5-amber-rgb)); }
.c-action { color: rgb(var(--v5-primary-rgb)); }
.c-flow { color: rgb(var(--v5-pink-rgb)); }
.c-extern { color: rgb(var(--v5-green-rgb)); }
.c-log { color: #6b6b76; }
.dark .c-log { color: #9aa0a6; }
/* ─── IDE takeover 5-분할 grid 골격 (V3Takeover 미러, 단계 2 부터 사용) ─── */
.ctrl-ide-shell {
position: absolute; inset: 0;
z-index: 25;
display: grid;
grid-template-columns: 248px 1fr 304px;
grid-template-rows: 44px 1fr 32px;
background: var(--v5-surface-solid);
border: 1px solid rgba(var(--v5-cyan-rgb), .3);
box-shadow:
0 0 0 4px rgba(var(--v5-cyan-rgb), .08),
0 30px 60px -20px rgba(0, 0, 0, .2);
animation: ctrl-ide-in .35s var(--v5-ease-enter) both;
}
@keyframes ctrl-ide-in {
from { opacity: 0; transform: translateY(8px); }
to { opacity: 1; transform: none; }
}
.ctrl-ide-ctxbar {
grid-column: 1 / span 3; grid-row: 1;
display: flex; align-items: center; gap: 10px;
padding: 0 .85rem;
background: var(--v5-surface-solid);
border-bottom: 1px solid var(--v5-border);
}
.ctrl-ide-leftrail {
grid-column: 1; grid-row: 2;
overflow: hidden;
display: flex; flex-direction: column;
background: var(--v5-surface-solid);
}
.ctrl-ide-canvas {
grid-column: 2; grid-row: 2;
position: relative;
border-left: 1px solid var(--v5-border);
border-right: 1px solid var(--v5-border);
overflow: hidden;
background: var(--v5-surface-solid);
}
.ctrl-ide-rightrail {
grid-column: 3; grid-row: 2;
overflow: hidden;
display: flex; flex-direction: column;
background: var(--v5-surface-solid);
}
.ctrl-ide-statusbar {
grid-column: 1 / span 3; grid-row: 3;
display: flex; align-items: center;
padding: 0 .85rem;
background: var(--v5-bg-subtle);
border-top: 1px solid var(--v5-border);
font-family: var(--v5-font-mono);
font-size: .62rem;
color: var(--v5-text-sec);
gap: 14px;
}
/* ContextBar — IDE 좌측 cyan glow 배지 */
.ctrl-ide-badge {
display: inline-flex; align-items: center; gap: 6px;
padding: 3px 9px;
background: linear-gradient(135deg,
rgba(var(--v5-cyan-rgb), .18),
rgba(var(--v5-primary-rgb), .10));
border: 1px solid rgba(var(--v5-cyan-rgb), .35);
color: rgb(0, 154, 150);
border-radius: 6px;
font-size: .62rem;
font-weight: 800;
letter-spacing: .06em;
font-family: var(--v5-font-mono);
text-transform: uppercase;
}
.dark .ctrl-ide-badge { color: rgb(var(--v5-cyan-rgb)); }
/* ContextBar — brumb separator */
.ctrl-ide-sep {
color: var(--v5-text-muted);
font-size: .7rem;
}
/* ContextBar — 모드 4-segmented tabs (READ/EDIT/RUN/HISTORY) */
.ctrl-ide-mode-tabs {
display: flex; gap: 2px;
padding: 2px;
background: var(--v5-bg-subtle);
border: 1px solid var(--v5-border);
border-radius: 7px;
}
.ctrl-ide-mode-tab {
padding: 4px 11px;
border: none;
background: transparent;
font-family: inherit;
font-size: .7rem;
font-weight: 600;
color: var(--v5-text-muted);
border-radius: 5px;
cursor: pointer;
display: inline-flex; align-items: center; gap: 5px;
transition: background .18s var(--v5-ease-move),
color .18s var(--v5-ease-move);
}
.ctrl-ide-mode-tab:hover { color: var(--v5-text-sec); }
.ctrl-ide-mode-tab.on {
background: var(--v5-surface-solid);
color: var(--v5-text);
box-shadow:
0 1px 2px rgba(0, 0, 0, .05),
0 0 0 1px var(--v5-border);
}
.ctrl-ide-mode-tab svg { width: 10px; height: 10px; stroke-width: 2; }
/* ContextBar — 우측 작은 툴 버튼 */
.ctrl-ide-tool {
display: inline-flex; align-items: center; gap: 5px;
padding: 4px 9px;
height: 26px;
border: 1px solid transparent;
background: transparent;
border-radius: 6px;
font-family: inherit;
font-size: .7rem;
font-weight: 600;
color: var(--v5-text-sec);
cursor: pointer;
transition: background .18s var(--v5-ease-move),
color .18s var(--v5-ease-move);
}
.ctrl-ide-tool:hover {
background: var(--v5-surface-hover);
color: var(--v5-text);
}
.ctrl-ide-tool.primary {
background: rgb(var(--v5-primary-rgb));
color: #fff;
font-weight: 700;
}
.ctrl-ide-tool.primary:hover { filter: brightness(1.08); }
.ctrl-ide-tool svg { width: 11px; height: 11px; stroke-width: 2; }
/* ─── Section Header — Rail 안 섹션 헤더 (v3 SectionHeader 미러) ─── */
.ctrl-sec-head {
display: flex; align-items: center;
padding: 8px 12px 6px;
font-size: .65rem;
font-weight: 700;
letter-spacing: .02em;
color: var(--v5-text-sec);
}
.ctrl-sec-head .ctrl-sec-ico {
margin-right: 6px;
display: inline-flex;
color: inherit;
}
.ctrl-sec-head .ctrl-sec-count {
margin-left: 4px;
font-family: var(--v5-font-mono);
font-size: .55rem;
color: var(--v5-text-muted);
font-weight: 600;
}
.ctrl-sec-head .ctrl-sec-right { margin-left: auto; }
/* ─── FAB — 카드 선택 전 (제어 ON 상태 안내 + 종료) (v3 .fab) ─── */
.ctrl-fab {
position: absolute;
bottom: 18px; right: 18px;
z-index: 40;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 999px;
padding: 6px 10px 6px 14px;
box-shadow:
0 0 16px rgba(var(--v5-cyan-rgb), .2),
0 10px 30px -10px rgba(0, 0, 0, .18);
display: flex; align-items: center; gap: 8px;
font-size: .72rem; font-weight: 600;
color: var(--v5-text);
animation: ctrl-fab-in .35s var(--v5-ease-enter) both;
}
.ctrl-fab .ctrl-fab-dot {
width: 7px; height: 7px;
border-radius: 50%;
background: rgb(var(--v5-cyan-rgb));
box-shadow: 0 0 8px rgb(var(--v5-cyan-rgb));
animation: ctrl-fab-pulse 1.4s infinite;
}
.ctrl-fab .ctrl-fab-sep {
width: 1px; height: 14px;
background: var(--v5-border);
}
.ctrl-fab .ctrl-fab-x {
width: 22px; height: 22px;
display: grid; place-items: center;
border: none;
background: var(--v5-bg-subtle);
border-radius: 6px;
cursor: pointer;
color: var(--v5-text-sec);
transition: background .18s var(--v5-ease-move),
color .18s var(--v5-ease-move);
}
.ctrl-fab .ctrl-fab-x:hover {
background: rgba(var(--v5-red-rgb), .1);
color: rgb(var(--v5-red-rgb));
}
.ctrl-fab .ctrl-fab-x svg { width: 11px; height: 11px; stroke-width: 2.5; }
@keyframes ctrl-fab-in {
from { opacity: 0; transform: translateY(12px); }
to { opacity: 1; transform: none; }
}
@keyframes ctrl-fab-pulse {
0%, 100% { opacity: 1; }
50% { opacity: .3; }
}
/* ─── Empty-state 안내 칩 — 제어 ON, 카드 미선택 (v3 V3Hint) ─── */
.ctrl-mode-hint {
position: absolute;
top: 18px; left: 50%;
transform: translateX(-50%);
z-index: 35;
display: inline-flex; align-items: center; gap: 10px;
padding: .55rem 1.1rem;
background: var(--v5-surface-solid);
border: 1px solid rgba(var(--v5-cyan-rgb), .35);
border-radius: 999px;
box-shadow:
0 0 0 4px rgba(var(--v5-cyan-rgb), .08),
0 10px 24px -8px rgba(var(--v5-cyan-rgb), .25);
font-size: .74rem;
font-weight: 600;
color: var(--v5-text);
white-space: nowrap;
pointer-events: none;
animation: ctrl-fab-in .35s var(--v5-ease-enter) both;
}
.ctrl-mode-hint b { color: rgb(0, 154, 150); }
.dark .ctrl-mode-hint b { color: rgb(var(--v5-cyan-rgb)); }
/* ─── Mini Topology Popover — 카드 호버 시 (v3 .dh-hpop) ─── */
.ctrl-hpop {
position: absolute;
z-index: 60;
width: 230px;
background: var(--v5-surface-solid);
border: 1px solid rgba(var(--v5-cyan-rgb), .35);
border-radius: 12px;
padding: .65rem .7rem;
box-shadow:
0 0 16px rgba(var(--v5-cyan-rgb), .2),
0 16px 40px -12px rgba(0, 0, 0, .15);
pointer-events: none;
animation: ctrl-hpop-in .22s var(--v5-ease-enter);
}
@keyframes ctrl-hpop-in {
from { opacity: 0; transform: translateY(-4px) scale(.96); }
to { opacity: 1; transform: translateY(0) scale(1); }
}
.ctrl-hpop .ctrl-hpop-l {
font-family: var(--v5-font-mono);
font-size: .55rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: .06em;
color: rgb(var(--v5-cyan-rgb));
margin-bottom: 4px;
display: flex; align-items: center; gap: 5px;
}
.ctrl-hpop .ctrl-hpop-l::before {
content: '';
width: 5px; height: 5px;
border-radius: 50%;
background: rgb(var(--v5-cyan-rgb));
box-shadow: 0 0 6px rgb(var(--v5-cyan-rgb));
animation: ctrl-fab-pulse 1.4s infinite;
}
.ctrl-hpop .ctrl-hpop-t {
font-size: .75rem;
font-weight: 700;
color: var(--v5-text);
margin-bottom: .3rem;
}
.ctrl-hpop .ctrl-hpop-svg {
width: 100%; height: 64px;
background: var(--v5-bg-subtle);
border-radius: 6px;
border: 1px solid var(--v5-border-subtle);
}
.ctrl-hpop .ctrl-hpop-stats {
display: flex; gap: .55rem;
margin-top: .4rem;
font-family: var(--v5-font-mono);
font-size: .6rem;
color: var(--v5-text-sec);
font-weight: 600;
}
.ctrl-hpop .ctrl-hpop-stats b {
color: var(--v5-text);
font-weight: 700;
}
/* ═══════════════════════════════════════════════════════════════════════════
IDE 컴포넌트 CSS — ContextBar / LeftRail / RightRail / Canvas / StatusBar
2026-05-19
═══════════════════════════════════════════════════════════════════════════ */
/* ─── ContextBar ─── */
.ctrl-ide-crumb-card {
display: inline-flex; align-items: center; gap: 6px;
font-size: .74rem; font-weight: 700;
color: var(--v5-text);
}
.ctrl-ide-crumb-tbl {
font-family: var(--v5-font-mono);
font-size: .6rem;
color: var(--v5-text-muted);
font-weight: 500;
margin-left: 4px;
}
.ctrl-ide-cmdk {
height: 22px;
padding: 0 7px;
font-family: var(--v5-font-mono);
font-size: .6rem;
}
.ctrl-ide-vsep {
width: 1px; height: 20px;
background: var(--v5-border);
flex-shrink: 0;
}
.ctrl-ide-close {
border: 1px solid var(--v5-border);
background: var(--v5-surface-solid);
}
.ctrl-ide-exit {
color: rgb(var(--v5-red-rgb)) !important;
}
.ctrl-ide-exit:hover {
background: rgba(var(--v5-red-rgb), .08) !important;
}
/* ─── LeftRail · 섹션 토글 ─── */
.ctrl-sec-toggle {
width: 100%;
border: none;
background: transparent;
cursor: pointer;
font-family: inherit;
}
.ctrl-sec-toggle:hover { background: var(--v5-surface-hover); }
/* ─── LeftRail · 카드 목록 ─── */
.ctrl-ide-cards {
display: flex; flex-direction: column;
gap: 2px;
padding: 0 6px 6px;
overflow-y: auto;
max-height: 30%;
border-bottom: 1px solid var(--v5-border-subtle);
}
.ctrl-ide-card-row {
display: flex; flex-direction: column;
gap: 2px;
width: 100%;
padding: 6px 8px;
border: 1px solid transparent;
background: transparent;
border-radius: 6px;
text-align: left;
cursor: pointer;
transition: background .15s var(--v5-ease-move), border-color .15s var(--v5-ease-move);
}
.ctrl-ide-card-row:hover {
background: var(--v5-surface-hover);
}
.ctrl-ide-card-row.on {
background: rgba(var(--v5-cyan-rgb), .08);
border-color: rgba(var(--v5-cyan-rgb), .35);
}
.ctrl-ide-card-row-title {
font-size: .72rem;
font-weight: 700;
color: var(--v5-text);
}
.ctrl-ide-card-row-tbl {
font-family: var(--v5-font-mono);
font-size: .56rem;
color: var(--v5-text-muted);
}
.ctrl-ide-empty {
padding: .8rem .7rem;
font-size: .62rem;
color: var(--v5-text-muted);
text-align: center;
font-style: italic;
}
/* ─── LeftRail · 노드 팔레트 ─── */
.ctrl-ide-palette-search {
display: flex; align-items: center;
gap: 5px;
margin: 6px 8px;
padding: 5px 8px;
background: var(--v5-bg-subtle);
border: 1px solid var(--v5-border);
border-radius: 6px;
color: var(--v5-text-muted);
}
.ctrl-ide-palette-search input {
flex: 1;
border: none;
background: transparent;
outline: none;
font-family: inherit;
font-size: .68rem;
color: var(--v5-text);
}
.ctrl-ide-palette {
flex: 1;
overflow-y: auto;
padding: 0 8px 8px;
}
.ctrl-ide-palette-cat {
margin-top: 8px;
}
.ctrl-ide-palette-cat-label {
padding: 4px 0;
}
.ctrl-ide-palette-items {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 4px;
margin-top: 4px;
}
.ctrl-ide-palette-item {
display: flex; align-items: center;
gap: 5px;
padding: 6px 7px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 6px;
cursor: grab;
font-family: inherit;
text-align: left;
transition: background .15s var(--v5-ease-move),
border-color .15s var(--v5-ease-move),
transform .12s var(--v5-ease-move);
}
.ctrl-ide-palette-item:hover {
background: var(--v5-surface-hover);
border-color: rgba(var(--v5-cyan-rgb), .35);
transform: translateY(-1px);
}
.ctrl-ide-palette-item:active { cursor: grabbing; }
.ctrl-ide-palette-ico {
font-size: .8rem;
width: 18px; height: 18px;
display: grid; place-items: center;
}
.ctrl-ide-palette-label {
font-size: .62rem;
font-weight: 600;
color: var(--v5-text);
}
/* ─── RightRail · Inspector ─── */
.ctrl-ide-inspector {
flex: 1;
overflow-y: auto;
padding: 0 12px 12px;
display: flex; flex-direction: column;
gap: 10px;
}
.ctrl-ide-field {
display: flex; flex-direction: column;
gap: 4px;
}
.ctrl-ide-field-label {
font-size: .62rem;
font-weight: 700;
color: var(--v5-text-sec);
letter-spacing: .01em;
}
.ctrl-ide-field-locked {
font-size: .56rem;
color: var(--v5-text-muted);
font-weight: 500;
}
.ctrl-ide-field-input {
width: 100%;
padding: 5px 8px;
border: 1px solid var(--v5-border);
background: var(--v5-surface-solid);
border-radius: 5px;
font-family: inherit;
font-size: .7rem;
color: var(--v5-text);
outline: none;
transition: border-color .15s var(--v5-ease-move),
box-shadow .15s var(--v5-ease-move);
}
.ctrl-ide-field-input:focus {
border-color: rgba(var(--v5-primary-rgb), .5);
box-shadow: 0 0 0 2px rgba(var(--v5-primary-rgb), .12);
}
.ctrl-ide-field-input:disabled {
background: var(--v5-bg-subtle);
color: var(--v5-text-muted);
cursor: not-allowed;
}
.ctrl-ide-field-input.mono {
font-family: var(--v5-font-mono);
}
.ctrl-ide-field-hint {
font-family: var(--v5-font-mono);
font-size: .58rem;
color: var(--v5-text-muted);
}
.ctrl-ide-field-meta {
display: flex; flex-direction: column;
gap: 5px;
padding: 8px 10px;
background: var(--v5-bg-subtle);
border-radius: 6px;
}
.ctrl-ide-field-meta > div {
display: flex; align-items: center;
gap: 8px;
font-size: .64rem;
}
.ctrl-ide-field-row {
display: flex; align-items: center;
gap: 8px;
padding: 6px 10px;
font-size: .68rem;
}
.ctrl-ide-field-row:nth-child(odd) { background: var(--v5-bg-subtle); }
.ctrl-ide-field-k {
font-family: var(--v5-font-mono);
font-size: .58rem;
font-weight: 700;
letter-spacing: .04em;
text-transform: uppercase;
color: var(--v5-text-muted);
min-width: 56px;
}
.ctrl-ide-card-info {
margin: 0 12px;
border: 1px solid var(--v5-border-subtle);
border-radius: 6px;
overflow: hidden;
}
.ctrl-ide-help {
padding: 0 12px;
font-size: .68rem;
color: var(--v5-text-sec);
line-height: 1.5;
}
.ctrl-ide-help p { margin: 6px 0; }
.ctrl-ide-help b { color: var(--v5-text); font-weight: 700; }
/* Inspector stats */
.ctrl-ide-stats {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 5px;
padding: 0 12px;
}
.ctrl-ide-stats > div {
display: flex; align-items: center;
gap: 6px;
padding: 6px 8px;
background: var(--v5-bg-subtle);
border-radius: 5px;
font-size: .64rem;
}
.ctrl-ide-stats code {
font-family: var(--v5-font-mono);
font-size: .66rem;
color: var(--v5-text);
font-weight: 700;
}
.ctrl-ide-stat-alert {
grid-column: 1 / span 2;
padding: 6px 8px;
background: rgba(var(--v5-amber-rgb), .12);
border: 1px solid rgba(var(--v5-amber-rgb), .35);
color: rgb(var(--v5-amber-rgb));
border-radius: 5px;
font-size: .62rem;
font-weight: 600;
}
/* Inspector comments */
.ctrl-ide-comments {
display: flex; flex-direction: column;
gap: 6px;
padding: 0 12px;
}
.ctrl-ide-comment {
display: grid;
grid-template-columns: 24px 1fr;
gap: 8px;
padding: 6px 8px;
background: var(--v5-bg-subtle);
border-radius: 5px;
}
.ctrl-ide-avatar {
width: 24px; height: 24px;
border-radius: 50%;
display: grid; place-items: center;
color: #fff;
font-size: .58rem;
font-weight: 700;
}
.ctrl-ide-comment-meta {
font-size: .58rem;
color: var(--v5-text-muted);
}
.ctrl-ide-comment-meta b { color: var(--v5-text); font-weight: 700; }
.ctrl-ide-comment-text {
font-size: .66rem;
color: var(--v5-text);
margin-top: 2px;
}
.ctrl-ide-mini {
height: 22px !important;
padding: 0 6px !important;
}
/* Inspector — validation dot */
.ctrl-validation-dot {
width: 7px; height: 7px;
border-radius: 50%;
display: inline-block;
}
.ctrl-validation-dot.ok { background: var(--v5-green); box-shadow: 0 0 6px var(--v5-green); }
.ctrl-validation-dot.bad { background: var(--v5-red); box-shadow: 0 0 6px var(--v5-red); }
/* ─── Canvas — 공통 ─── */
.ctrl-ide-canvas-inner {
position: absolute; inset: 0;
}
/* EditCanvas → RuleBuilder host (RuleBuilder 는 position:absolute, inset:0) */
.ctrl-edit-canvas-host {
position: absolute; inset: 0;
overflow: hidden;
}
/* RuleBuilder 캔버스 — v3 EditCanvas mockup 의 dot 패턴 배경 (solid + 보라톤 도트) */
.rule-builder-canvas {
background-color: var(--v5-surface-solid);
background-image: radial-gradient(circle, rgba(var(--v5-primary-rgb), .16) 0.8px, transparent 1.2px);
background-size: 16px 16px;
background-position: 0 0;
}
.ctrl-canvas-tag {
font-family: var(--v5-font-mono);
font-size: .55rem;
font-weight: 700;
letter-spacing: .08em;
text-transform: uppercase;
}
.ctrl-canvas-mono {
font-family: var(--v5-font-mono);
font-size: .6rem;
color: var(--v5-text-muted);
}
.ctrl-canvas-relnode {
background: var(--v5-surface-solid);
border-width: 1.5px;
border-style: solid;
border-radius: 10px;
padding: .55rem .7rem;
box-shadow: 0 6px 16px -8px rgba(0, 0, 0, .10);
}
.ctrl-canvas-empty {
position: absolute; inset: 0;
display: flex; flex-direction: column;
align-items: center; justify-content: center;
gap: 4px;
color: var(--v5-text-muted);
font-size: .8rem;
font-style: italic;
}
.ctrl-canvas-empty small {
font-family: var(--v5-font-mono);
font-size: .58rem;
font-style: normal;
}
/* ─── PanZoomStage ─── */
.ctrl-pz-stage {
position: absolute; inset: 0;
overflow: hidden;
}
.ctrl-pz-zoom {
position: absolute; bottom: 12px; right: 12px;
z-index: 10;
display: flex; gap: 4px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 7px;
padding: 3px;
box-shadow: 0 2px 6px rgba(0, 0, 0, .06);
}
.ctrl-pz-zoom button {
width: 24px; height: 24px;
border: none;
background: transparent;
border-radius: 5px;
display: grid; place-items: center;
cursor: pointer;
color: var(--v5-text-sec);
transition: background .15s var(--v5-ease-move), color .15s var(--v5-ease-move);
}
.ctrl-pz-zoom button:hover {
background: var(--v5-surface-hover);
color: var(--v5-text);
}
.ctrl-pz-pct {
padding: 0 7px;
font-family: var(--v5-font-mono);
font-size: .6rem;
font-weight: 700;
color: var(--v5-text-sec);
display: inline-flex; align-items: center;
}
.ctrl-pz-hint {
position: absolute; bottom: 12px; left: 12px;
z-index: 10;
padding: 4px 9px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 999px;
font-family: var(--v5-font-mono);
font-size: .58rem;
font-weight: 600;
color: var(--v5-text-muted);
display: inline-flex; align-items: center;
gap: 5px;
}
/* ─── Run mode ─── */
.ctrl-run-shell {
position: absolute; inset: 0;
display: flex; flex-direction: column;
}
.ctrl-run-top {
padding: .8rem 1rem;
background: linear-gradient(135deg, rgba(var(--v5-cyan-rgb), .08), rgba(var(--v5-primary-rgb), .05));
border-bottom: 1px solid var(--v5-border);
display: flex; align-items: center; gap: 14px;
}
.ctrl-run-state {
width: 36px; height: 36px;
border-radius: 9px;
color: #fff;
display: grid; place-items: center;
}
.ctrl-run-state.playing {
background: var(--v5-green);
box-shadow: 0 0 20px rgba(var(--v5-green-rgb), .4);
}
.ctrl-run-state.paused {
background: var(--v5-amber);
box-shadow: 0 0 20px rgba(var(--v5-amber-rgb), .4);
}
.ctrl-canvas-tag.is-play { color: var(--v5-green); }
.ctrl-canvas-tag.is-pause { color: var(--v5-amber); }
.ctrl-run-btns {
display: flex; gap: 3px;
padding: 3px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 8px;
}
.ctrl-run-btn {
width: 28px; height: 24px;
border: none;
background: transparent;
color: var(--v5-text-sec);
border-radius: 5px;
cursor: pointer;
display: grid; place-items: center;
}
.ctrl-run-btn:hover { background: var(--v5-surface-hover); color: var(--v5-text); }
.ctrl-run-btn.primary {
background: rgb(var(--v5-primary-rgb));
color: #fff;
box-shadow: var(--v5-glow-sm);
}
.ctrl-run-counter {
text-align: right;
min-width: 80px;
}
.ctrl-run-counter-num {
font-family: var(--v5-font-mono);
font-size: 1.05rem;
font-weight: 800;
color: var(--v5-text);
letter-spacing: -.02em;
}
.ctrl-run-progress {
height: 3px;
background: var(--v5-bg-subtle);
position: relative; overflow: hidden;
}
.ctrl-run-progress-bar {
position: absolute; left: 0; top: 0; bottom: 0;
background: linear-gradient(90deg, rgba(var(--v5-cyan-rgb), .85), rgba(var(--v5-primary-rgb), .85));
transition: width .5s ease-out;
}
.ctrl-run-steps {
flex: 1; overflow: auto;
padding: .9rem 1rem;
display: flex; flex-direction: column;
gap: 7px;
}
.ctrl-run-step {
display: grid;
grid-template-columns: 28px 24px 1fr auto auto;
gap: 10px;
align-items: center;
padding: 9px 11px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border-subtle);
border-radius: 8px;
transition: background .2s var(--v5-ease-move), border-color .2s var(--v5-ease-move);
}
.ctrl-run-step.is-active {
background: rgba(var(--v5-cyan-rgb), .06);
border-color: rgba(var(--v5-cyan-rgb), .4);
box-shadow: 0 0 0 3px rgba(var(--v5-cyan-rgb), .08);
}
.ctrl-run-step.is-pending { opacity: .45; }
.ctrl-run-step-num {
width: 22px; height: 22px;
border-radius: 50%;
display: grid; place-items: center;
font-family: var(--v5-font-mono);
font-size: .6rem;
font-weight: 800;
}
.ctrl-run-step-ico {
width: 22px; height: 22px;
border-radius: 6px;
display: grid; place-items: center;
font-size: .8rem;
}
.ctrl-run-step-status {
font-family: var(--v5-font-mono);
font-size: .58rem;
color: var(--v5-text-muted);
min-width: 70px;
text-align: right;
}
.ctrl-run-step-status.is-active {
color: rgb(0, 154, 150);
font-weight: 700;
}
.dark .ctrl-run-step-status.is-active { color: rgb(var(--v5-cyan-rgb)); }
/* Latency bar */
.ctrl-latency-bar {
display: inline-flex; align-items: center;
gap: 6px;
width: 70px;
height: 5px;
background: var(--v5-bg-subtle);
border-radius: 999px;
position: relative; overflow: visible;
}
.ctrl-latency-bar > div {
height: 100%;
border-radius: 999px;
transition: width .3s var(--v5-ease-move);
}
.ctrl-latency-bar > span {
position: absolute;
right: -42px;
top: -5px;
font-family: var(--v5-font-mono);
font-size: .56rem;
color: var(--v5-text-muted);
font-weight: 700;
}
/* ─── History mode ─── */
.ctrl-history-shell {
position: absolute; inset: 0;
display: flex; flex-direction: column;
}
.ctrl-history-top {
padding: .65rem .9rem;
border-bottom: 1px solid var(--v5-border);
display: flex; align-items: center;
gap: 14px;
}
.ctrl-history-tag {
display: inline-flex; align-items: center;
gap: 6px;
padding: 3px 8px;
border-radius: 6px;
background: var(--v5-bg-subtle);
border: 1px solid var(--v5-border);
font-family: var(--v5-font-mono);
font-size: .62rem;
font-weight: 700;
letter-spacing: .04em;
text-transform: uppercase;
color: var(--v5-text-sec);
}
.ctrl-history-select {
height: 26px;
padding: 0 .4rem;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 5px;
font-family: inherit;
font-size: .66rem;
color: var(--v5-text);
}
.ctrl-history-body {
flex: 1; overflow: auto;
}
.ctrl-history-table {
width: 100%;
border-collapse: collapse;
font-size: .7rem;
}
.ctrl-history-table thead th {
padding: .55rem .85rem;
text-align: left;
font-family: var(--v5-font-mono);
font-size: .54rem;
font-weight: 700;
letter-spacing: .06em;
text-transform: uppercase;
color: var(--v5-text-muted);
background: var(--v5-bg-subtle);
border-bottom: 1px solid var(--v5-border);
position: sticky; top: 0;
z-index: 1;
}
.ctrl-history-table tbody td {
padding: .55rem .85rem;
border-bottom: 1px solid var(--v5-border-subtle);
vertical-align: middle;
}
.ctrl-history-mono {
font-family: var(--v5-font-mono);
font-size: .66rem;
font-weight: 600;
color: var(--v5-text);
}
.ctrl-history-sec { color: var(--v5-text-sec) !important; }
.ctrl-history-dot {
width: 7px; height: 7px;
display: inline-block;
border-radius: 50%;
}
.ctrl-history-result {
padding: 1.5px 8px;
border-radius: 999px;
font-family: var(--v5-font-mono);
font-size: .58rem;
font-weight: 800;
letter-spacing: .04em;
text-transform: uppercase;
}
.ctrl-history-result.ok {
background: rgba(var(--v5-green-rgb), .1);
color: var(--v5-green);
}
.ctrl-history-result.fail {
background: rgba(var(--v5-red-rgb), .1);
color: var(--v5-red);
}
.ctrl-history-more {
width: 22px; height: 22px;
border: none;
background: transparent;
border-radius: 5px;
cursor: pointer;
color: var(--v5-text-muted);
}
.ctrl-history-more:hover {
background: var(--v5-surface-hover);
color: var(--v5-text);
}
/* ─── StatusBar ─── */
.ctrl-ide-statusbar code {
font-family: var(--v5-font-mono);
font-size: .62rem;
color: var(--v5-text);
font-weight: 700;
}
.ctrl-ide-statusbar b {
font-family: var(--v5-font-mono);
font-size: .55rem;
font-weight: 800;
letter-spacing: .06em;
text-transform: uppercase;
color: var(--v5-text-muted);
margin-right: 4px;
}
.ctrl-ide-status-ok {
color: var(--v5-green);
font-family: var(--v5-font-mono);
font-size: .6rem;
font-weight: 700;
}
/* ═══════════════════════════════════════════════════════════════════════════
LeftRail v2 — v3 V3LeftRail + invyone 테이블 팔레트 통합
═══════════════════════════════════════════════════════════════════════════ */
/* 검색 박스 */
.ctrl-rail-search {
display: flex; align-items: center; gap: 5px;
margin: 8px 8px 6px;
padding: 5px 8px;
background: var(--v5-bg-subtle);
border: 1px solid var(--v5-border);
border-radius: 6px;
color: var(--v5-text-muted);
flex-shrink: 0;
}
.ctrl-rail-search input {
flex: 1;
border: none;
background: transparent;
outline: none;
font-family: inherit;
font-size: .68rem;
color: var(--v5-text);
}
.ctrl-rail-search input::placeholder { color: var(--v5-text-muted); }
/* 섹션 — V3RailSection 미러 */
.ctrl-rail-sec {
display: flex; flex-direction: column;
flex: 0 0 auto;
min-height: 0;
border-bottom: 1px solid var(--v5-border-subtle);
}
.ctrl-rail-sec.expand { flex: 1 1 0; }
.ctrl-rail-sec-head {
padding: .55rem .7rem;
display: flex; align-items: center; gap: 6px;
background: var(--v5-bg-subtle);
font-family: var(--v5-font-mono);
font-size: .58rem; font-weight: 700;
letter-spacing: .06em; text-transform: uppercase;
color: var(--v5-text-sec);
flex-shrink: 0;
}
.ctrl-rail-sec-title { flex: 1; }
.ctrl-rail-sec-count { color: var(--v5-text-muted); }
.ctrl-rail-sec-body {
overflow: auto;
flex: 1; min-height: 0;
}
/* 카드 목록 */
.ctrl-rail-cards {
padding: 5px 7px;
display: flex; flex-direction: column;
gap: 3px;
}
.ctrl-rail-card {
display: flex; align-items: center; gap: 7px;
padding: 6px 8px;
background: transparent;
border: 1px solid transparent;
border-radius: 7px;
cursor: pointer;
text-align: left;
font-family: inherit;
width: 100%;
transition: background .15s var(--v5-ease-move), border-color .15s var(--v5-ease-move);
}
.ctrl-rail-card:hover { background: var(--v5-surface-hover); }
.ctrl-rail-card.on {
background: rgba(var(--v5-cyan-rgb), .12);
border-color: rgba(var(--v5-cyan-rgb), .35);
}
.ctrl-rail-card-ico {
width: 22px; height: 22px;
border-radius: 5px;
background: var(--v5-bg-subtle);
color: var(--v5-text-sec);
display: grid; place-items: center;
flex-shrink: 0;
}
.ctrl-rail-card.on .ctrl-rail-card-ico {
background: rgba(var(--v5-cyan-rgb), .2);
color: rgb(0, 154, 150);
}
.dark .ctrl-rail-card.on .ctrl-rail-card-ico { color: rgb(var(--v5-cyan-rgb)); }
.ctrl-rail-card-body { flex: 1; min-width: 0; }
.ctrl-rail-card-title {
font-size: .72rem; font-weight: 600;
color: var(--v5-text);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.ctrl-rail-card-tbl {
font-family: var(--v5-font-mono);
font-size: .58rem;
color: var(--v5-text-muted);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.ctrl-rail-card-chev { color: rgb(0, 154, 150); flex-shrink: 0; }
.dark .ctrl-rail-card-chev { color: rgb(var(--v5-cyan-rgb)); }
/* 카테고리 라벨 — v3 V3LeftRail 미러 (mono uppercase + dot) */
.ctrl-rail-cat-label {
display: flex; align-items: center; gap: 5px;
margin-bottom: 4px;
padding: 0 2px;
font-family: var(--v5-font-mono);
font-size: .55rem;
font-weight: 700;
letter-spacing: .06em;
text-transform: uppercase;
}
.ctrl-rail-cat-dot {
width: 6px; height: 6px;
border-radius: 2px;
}
.ctrl-rail-cat-count {
color: var(--v5-text-muted);
font-weight: 500;
}
.ctrl-rec-label { color: var(--v5-amber); }
/* 테이블 — invyone 기존 디자인 유지 */
.ctrl-rail-tbls-wrap {
padding: 6px 7px;
}
.ctrl-rail-tbls-others { padding-top: 0; }
.ctrl-rail-tbls {
display: flex; flex-direction: column;
gap: 3px;
}
.ctrl-rail-tbl {
display: flex; align-items: center; gap: 6px;
padding: 5px 7px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 6px;
cursor: grab;
position: relative;
transition: background .15s var(--v5-ease-move),
border-color .15s var(--v5-ease-move);
}
.ctrl-rail-tbl:hover {
background: var(--v5-surface-hover);
border-color: rgba(var(--v5-cyan-rgb), .35);
}
.ctrl-rail-tbl:active { cursor: grabbing; }
.ctrl-rail-tbl.rec {
border-color: rgba(var(--v5-amber-rgb), .35);
background: rgba(var(--v5-amber-rgb), .04);
}
.ctrl-rail-tbl-ico {
font-size: .85rem;
width: 16px; text-align: center;
flex-shrink: 0;
}
.ctrl-rail-tbl-main {
flex: 1; min-width: 0;
display: flex; flex-direction: column;
}
.ctrl-rail-tbl-label {
font-size: .65rem;
font-weight: 600;
color: var(--v5-text);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.ctrl-rail-tbl-sub {
font-family: var(--v5-font-mono);
font-size: .53rem;
color: var(--v5-text-muted);
white-space: nowrap; overflow: hidden; text-overflow: ellipsis;
}
.ctrl-rail-tbl-star {
color: var(--v5-amber);
fill: var(--v5-amber);
flex-shrink: 0;
}
/* 노드 팔레트 */
.ctrl-rail-nodes {
padding: 6px 7px;
}
.ctrl-rail-nodecat {
margin-bottom: 10px;
}
.ctrl-rail-nodes-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 3px;
}
.ctrl-rail-node {
display: flex; align-items: center; gap: 4px;
padding: 5px 6px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 6px;
cursor: grab;
font-family: inherit;
font-size: .6rem; font-weight: 600;
color: var(--v5-text);
text-align: left;
transition: background .15s var(--v5-ease-move),
border-color .15s var(--v5-ease-move),
transform .12s var(--v5-ease-move);
}
.ctrl-rail-node:hover {
background: var(--v5-surface-hover);
border-color: rgba(var(--v5-cyan-rgb), .35);
transform: translateY(-1px);
}
.ctrl-rail-node:active { cursor: grabbing; }
.ctrl-rail-node-ico {
font-size: .75rem;
width: 14px; text-align: center;
flex-shrink: 0;
}
.ctrl-rail-node-label {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.ctrl-rail-empty {
padding: .6rem .5rem;
font-size: .62rem;
color: var(--v5-text-muted);
text-align: center;
font-style: italic;
}
.ctrl-rail-hint {
margin: 12px;
padding: 10px 12px;
background: var(--v5-bg-subtle);
border: 1px dashed var(--v5-border);
border-radius: 7px;
display: flex; align-items: center;
gap: 8px;
font-size: .65rem;
color: var(--v5-text-muted);
line-height: 1.4;
}
/* ═══════════════════════════════════════════════════════════════════════════
V3RuleNode — v3-canvas.jsx V3RuleNode 정확 포팅 (cat-stripe + validation dot + comment + cat-chip + label + summary + stats + ports)
═══════════════════════════════════════════════════════════════════════════ */
.v3-rule-node {
position: absolute;
width: 150px;
background: var(--v5-surface-solid);
border: 1.5px solid;
border-radius: 9px;
cursor: grab;
overflow: visible;
user-select: none;
touch-action: none;
pointer-events: auto;
transition: opacity .15s var(--v5-ease-move),
box-shadow .2s var(--v5-ease-move),
border-color .2s var(--v5-ease-move);
}
/* 자식 (stripe / vdot / comment / body / port) 가 pointer 이벤트 안 흡수 — V3RuleNode div 가 받음 */
.v3-rule-node-stripe,
.v3-rule-node-vdot,
.v3-rule-node-comment,
.v3-rule-node-body,
.v3-rule-node-port { pointer-events: none; }
.v3-rule-node:active { cursor: grabbing; }
.v3-rule-node.is-dim { opacity: .4; }
.v3-rule-node-stripe {
height: 3px;
opacity: .9;
border-radius: 9px 9px 0 0;
}
.v3-rule-node-vdot {
position: absolute;
top: -4px; right: -4px;
width: 11px; height: 11px;
border-radius: 50%;
border: 2px solid var(--v5-surface-solid);
}
.v3-rule-node-comment {
position: absolute;
top: -7px; left: -6px;
z-index: 3;
width: 16px; height: 16px;
border-radius: 50%;
display: grid; place-items: center;
color: #fff;
font-size: .55rem;
font-weight: 700;
border: 2px solid var(--v5-surface-solid);
}
.v3-rule-node-body { padding: .45rem .6rem .5rem; }
.v3-rule-node-cat {
display: flex; align-items: center; gap: 6px;
margin-bottom: 4px;
}
.v3-rule-node-cat-ico {
width: 20px; height: 20px;
border-radius: 5px;
display: grid; place-items: center;
flex-shrink: 0;
}
.v3-rule-node-cat-label {
font-size: .53rem;
font-weight: 700;
letter-spacing: .05em;
text-transform: uppercase;
font-family: var(--v5-font-mono);
}
.v3-rule-node-label {
font-size: .73rem;
font-weight: 700;
color: var(--v5-text);
letter-spacing: -.01em;
margin-bottom: 3px;
}
.v3-rule-node-summary {
font-family: var(--v5-font-mono);
font-size: .55rem;
color: var(--v5-text-muted);
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.v3-rule-node-stats {
margin-top: 4px;
padding-top: 4px;
border-top: 1px dashed var(--v5-border);
display: flex;
justify-content: space-between;
font-family: var(--v5-font-mono);
font-size: .5rem;
color: var(--v5-text-muted);
}
.v3-rule-node-port {
position: absolute;
width: 8px; height: 8px;
border-radius: 50%;
top: 24px;
}
.v3-rule-node-port-in {
left: -4px;
background: var(--v5-surface-solid);
border-width: 1.5px;
border-style: solid;
}
.v3-rule-node-port-out {
right: -4px;
}
/* ─── EditCanvas title chip ─── */
.v3-edit-title-chip {
position: absolute;
top: 14px; left: 14px;
display: inline-flex; align-items: center; gap: 8px;
padding: 5px 11px;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 7px;
font-size: .7rem;
font-weight: 700;
color: var(--v5-text);
box-shadow: 0 1px 2px rgba(0, 0, 0, .04);
z-index: 5;
}
.v3-edit-title-ver {
font-family: var(--v5-font-mono);
font-size: .6rem;
color: var(--v5-text-muted);
margin-left: 4px;
}
/* ─── Context Menu (우클릭) ─── */
.v3-ctxmenu {
position: absolute;
background: var(--v5-surface-solid);
border: 1px solid var(--v5-border);
border-radius: 8px;
box-shadow:
0 12px 28px -8px rgba(0, 0, 0, .18),
0 2px 4px rgba(0, 0, 0, .04);
min-width: 180px;
padding: 4px;
z-index: 30;
}
.v3-ctxmenu-item {
width: 100%;
display: flex; align-items: center; gap: 8px;
padding: 6px 9px;
border: none;
background: transparent;
border-radius: 5px;
font-family: inherit;
font-size: .7rem;
font-weight: 500;
color: var(--v5-text);
cursor: pointer;
text-align: left;
transition: background .15s var(--v5-ease-move);
}
.v3-ctxmenu-item svg { color: var(--v5-text-sec); flex-shrink: 0; }
.v3-ctxmenu-item:hover { background: var(--v5-surface-hover); }
.v3-ctxmenu-item.danger { color: var(--v5-red); }
.v3-ctxmenu-item.danger svg { color: var(--v5-red); }
.v3-ctxmenu-sep {
height: 1px;
background: var(--v5-border);
margin: 4px 0;
}
/* ═══════════════════════════════════════════════════════════════════════════
ContextBar Presence Stack
═══════════════════════════════════════════════════════════════════════════ */
.ctrl-presence {
display: inline-flex; align-items: center;
margin-right: 4px;
}
.ctrl-presence-avatar {
width: 20px; height: 20px;
border-radius: 50%;
border: 2px solid var(--v5-surface-solid);
margin-left: -6px;
display: grid; place-items: center;
color: #fff;
font-size: .55rem;
font-weight: 700;
position: relative;
}
.ctrl-presence-avatar:first-child { margin-left: 0; }
.ctrl-presence-avatar.is-edit::after {
content: '';
position: absolute;
bottom: -1px; right: -1px;
width: 6px; height: 6px;
border-radius: 50%;
background: var(--v5-green);
border: 1.5px solid var(--v5-surface-solid);
}
.ctrl-presence-more {
width: 20px; height: 20px;
margin-left: -6px;
border-radius: 50%;
background: var(--v5-bg-subtle);
border: 2px solid var(--v5-surface-solid);
color: var(--v5-text-sec);
display: grid; place-items: center;
font-family: var(--v5-font-mono);
font-size: .52rem;
font-weight: 700;
}
/* ═══════════════════════════════════════════════════════════════════════════
RightRail Activity Section (실행 상태 · live)
═══════════════════════════════════════════════════════════════════════════ */
.ctrl-activity {
padding: .65rem .7rem;
display: flex; flex-direction: column;
gap: 7px;
}
.ctrl-activity-row {
display: flex; align-items: center; justify-content: space-between;
padding: 5px 8px;
background: var(--v5-bg-subtle);
border-radius: 5px;
}
.ctrl-activity-label {
font-family: var(--v5-font-mono);
font-size: .55rem;
font-weight: 700;
letter-spacing: .04em;
text-transform: uppercase;
color: var(--v5-text-muted);
}
.ctrl-activity-value {
display: inline-flex; align-items: center; gap: 5px;
font-size: .68rem;
font-weight: 600;
color: var(--v5-text);
}
.ctrl-activity-dot {
width: 6px; height: 6px;
border-radius: 50%;
}
.ctrl-activity-dot.ok { background: var(--v5-green); box-shadow: 0 0 5px var(--v5-green); }
.ctrl-activity-dot.warn { background: var(--v5-amber); box-shadow: 0 0 5px var(--v5-amber); }
.ctrl-activity-dot.bad { background: var(--v5-red); box-shadow: 0 0 5px var(--v5-red); }
/* ═══════════════════════════════════════════════════════════════════════════
StatusBar v2 — v3 V3StatusBar 미러 (Workflow icon + RULE_NAME + 진행 dot + 라텐시 + 최근 실행)
═══════════════════════════════════════════════════════════════════════════ */
.ctrl-status-rule {
display: inline-flex; align-items: center; gap: 5px;
font-weight: 700;
color: var(--v5-text);
font-family: var(--v5-font-sans);
font-size: .68rem;
}
.ctrl-status-ver {
font-family: var(--v5-font-mono);
font-size: .55rem;
color: var(--v5-text-muted);
}
.ctrl-status-pulse {
width: 6px; height: 6px;
border-radius: 50%;
background: var(--v5-green);
box-shadow: 0 0 6px var(--v5-green);
animation: ctrl-fab-pulse 1.4s infinite;
}
/* ═══════════════════════════════════════════════════════════════════════════
헤더 아래 cyan flash glow — v3 .dh-glow.on 미러
═══════════════════════════════════════════════════════════════════════════ */
.v5-hdr::after {
content: '';
position: absolute;
left: 0; right: 0; bottom: -1px;
height: 1px;
background: linear-gradient(90deg,
transparent,
rgb(var(--v5-cyan-rgb)),
rgb(var(--v5-green-rgb)),
rgb(var(--v5-cyan-rgb)),
transparent);
opacity: 0;
transition: opacity .4s var(--v5-ease-move);
pointer-events: none;
}
body:has(.ud-htool[data-mode="ctrl"].on) .v5-hdr::after {
opacity: 1;
animation: ctrl-dh-glow-pulse 2.4s ease-in-out infinite;
}
@keyframes ctrl-dh-glow-pulse {
0%, 100% { opacity: .4; }
50% { opacity: 1; }
}