/* AGX Node Graph v5 — T1/T2, Cost w/ sub-items, progress bars */
#nodeGraphTab{display:none;position:fixed;left:0;right:0;bottom:0;z-index:99;background:var(--ng-bg,#0b0e16);overflow:hidden;font-family:'Segoe UI',system-ui,sans-serif}
#nodeGraphTab.active{display:flex;flex-direction:column}
/* Maximize toggle — hide the AGX nav header AND the job-detail
   metrics strip + sub-tab nav so the graph claims the full
   viewport. The graph is z-index 99 and (when maximized) pinned
   top:0 → it overlaps anything underneath, but we still need to
   collapse the page elements above it so the layout doesn't push
   the graph back down. */
body.ng-graph-fullscreen #app-container > header { display: none !important; }
body.ng-graph-fullscreen #wip-job-detail-view .jh-metrics-strip { display: none !important; }
body.ng-graph-fullscreen #wip-job-detail-view > .ws-right-tabs,
body.ng-graph-fullscreen #wip-job-detail-view .ws-col-right > .ws-right-tabs {
  display: none !important;
}
body.ng-graph-fullscreen #nodeGraphTab.active {
  top: 0 !important;
  height: 100vh;
}
/* Theme vars */
:root{--ng-bg:#0b0e16;--ng-surface:#12162a;--ng-surface2:#10141e;--ng-border:#242e50;--ng-border2:#1c2236;--ng-text:#e4e6f0;--ng-dim:#6a7090;--ng-dim2:#4a5068;--ng-input:#0d1019;--ng-grid-dot:#161b2a;--ng-grid-line:#141828;--ng-port-bg:#0b0e16;--ng-shadow:rgba(0,0,0,.5)}
body.light-mode{--ng-bg:#f7f8fa;--ng-surface:#ffffff;--ng-surface2:#eceef2;--ng-border:#6b7280;--ng-border2:#9ca3af;--ng-text:#1f2937;--ng-dim:#4b5563;--ng-dim2:#6b7280;--ng-input:#f9fafb;--ng-grid-dot:#dde0e6;--ng-grid-line:#eef0f3;--ng-port-bg:#f7f8fa;--ng-shadow:rgba(0,0,0,.12)}

/* Light-mode overrides for hardcoded dark-mode colors */
body.light-mode .ng-node{border-width:1.5px}
body.light-mode .ng-node:hover{border-color:#1f2937}
body.light-mode .ng-hdr-name{text-shadow:none}
body.light-mode .ng-t-t1 .ng-hdr{background:linear-gradient(135deg,rgba(37,99,235,.55),rgba(37,99,235,.18));color:#1e3a8a;box-shadow:inset 0 0 18px rgba(37,99,235,.22)}
body.light-mode .ng-t-t2 .ng-hdr{background:linear-gradient(135deg,rgba(5,150,105,.55),rgba(5,150,105,.18));color:#064e3b;box-shadow:inset 0 0 18px rgba(5,150,105,.22)}
body.light-mode .ng-t-cost .ng-hdr{background:linear-gradient(135deg,rgba(217,119,6,.55),rgba(217,119,6,.18));color:#78350f;box-shadow:inset 0 0 18px rgba(217,119,6,.22)}
body.light-mode .ng-t-sub .ng-hdr{background:linear-gradient(135deg,rgba(124,58,237,.55),rgba(124,58,237,.18));color:#4c1d95;box-shadow:inset 0 0 18px rgba(124,58,237,.22)}
body.light-mode .ng-tt-po .ng-hdr{background:linear-gradient(135deg,rgba(8,145,178,.5),rgba(8,145,178,.18));color:#155e75;box-shadow:inset 0 0 18px rgba(8,145,178,.22)}
body.light-mode .ng-tt-inv .ng-hdr{background:linear-gradient(135deg,rgba(251,207,232,.7),rgba(251,207,232,.3));color:#9d174d;box-shadow:inset 0 0 18px rgba(251,207,232,.4)}
body.light-mode .ng-t-co .ng-hdr{background:linear-gradient(135deg,rgba(220,38,38,.5),rgba(219,39,119,.22));color:#831843;box-shadow:inset 0 0 18px rgba(219,39,119,.22)}
body.light-mode .ng-t-math .ng-hdr{background:linear-gradient(135deg,rgba(79,70,229,.55),rgba(79,70,229,.18));color:#312e81;box-shadow:inset 0 0 18px rgba(79,70,229,.22)}
body.light-mode .ng-t-watch .ng-hdr{background:linear-gradient(135deg,rgba(217,119,6,.5),rgba(219,39,119,.25));color:#78350f;box-shadow:inset 0 0 20px rgba(217,119,6,.28)}
body.light-mode .ng-t-note .ng-hdr{background:linear-gradient(135deg,rgba(75,85,99,.4),rgba(75,85,99,.15));color:#1f2937;box-shadow:inset 0 0 16px rgba(75,85,99,.18)}
body.light-mode .ng-t-job .ng-hdr{background:linear-gradient(135deg,rgba(37,99,235,.6),rgba(37,99,235,.22));color:#1e3a8a;box-shadow:inset 0 0 20px rgba(37,99,235,.28)}
body.light-mode .ng-t-wip .ng-hdr{background:linear-gradient(135deg,rgba(217,119,6,.5),rgba(219,39,119,.28));color:#78350f;box-shadow:inset 0 0 20px rgba(219,39,119,.25)}
body.light-mode .ng-wip-chip{color:#b45309}
body.light-mode .ng-pv{color:#374151}
body.light-mode .ng-si-sm{color:#374151}
body.light-mode .ng-si-val{color:#b45309}
body.light-mode .ng-sub-total{color:#b45309}
body.light-mode .ng-pct-val{color:#b45309}
body.light-mode .ng-coll-val{color:#1f2937}
body.light-mode .ng-coll-pct{color:#b45309}
body.light-mode .ng-coll-lbl{color:#4b5563}
body.light-mode .ng-coll-sep{color:#9ca3af}
body.light-mode .ng-cv-bud{color:#6d28d9}
body.light-mode .ng-cv-grn{color:#047857}
body.light-mode .ng-cv-yel{color:#b45309}
body.light-mode .ng-wv.ng-vp{color:#047857}
body.light-mode .ng-wv.ng-vn{color:#b91c1c}
body.light-mode .ng-wv-label{color:#b45309}
body.light-mode .ng-p{border-color:#9ca3af}
body.light-mode .ng-del-overlay{background:rgba(0,0,0,.35)}
body.light-mode .ng-del-box{background:#ffffff;border-color:#d1d5db}
body.light-mode .ng-del-title{color:#111827}
body.light-mode .ng-del-sub{color:#4b5563}
body.light-mode .ng-del-remove{background:#2563eb;color:#fff}
body.light-mode .ng-del-remove:hover{background:#1d4ed8}
body.light-mode .ng-del-permanent{background:#dc2626;color:#fff}
body.light-mode .ng-del-permanent:hover{background:#b91c1c}
body.light-mode .ng-del-cancel{background:#f3f4f6;color:#374151;border-color:#d1d5db}
body.light-mode .ng-del-cancel:hover{background:#e5e7eb}
/* Top bar */
.ng-topbar{display:flex;align-items:center;gap:8px;padding:6px 14px;background:var(--ng-surface2);border-bottom:1px solid var(--ng-border2);flex-shrink:0;z-index:20}
.ng-topbar-title{font-size:14px;font-weight:700;color:var(--ng-text)}
.ng-topbar-sub{font-size:11px;color:var(--ng-dim2)}
.ng-topbar-spacer{flex:1}
.ng-save-indicator{display:inline-flex;align-items:center;gap:5px;padding:3px 10px;font-size:11px;font-weight:600;color:var(--ng-dim,#8b90a5);border-radius:10px;background:rgba(255,255,255,0.03);border:1px solid var(--ng-border2,#1c2236);transition:color .2s,background .2s,border-color .2s}
.ng-save-dot{font-weight:700;font-size:13px;line-height:1}
.ng-save-indicator.ng-save-saving{color:#60a5fa;background:rgba(96,165,250,.08);border-color:rgba(96,165,250,.25)}
.ng-save-indicator.ng-save-saved{color:#34d399;background:rgba(52,211,153,.08);border-color:rgba(52,211,153,.25)}
.ng-save-indicator.ng-save-error{color:#f87171;background:rgba(248,113,113,.08);border-color:rgba(248,113,113,.3)}
.ng-tbtn{background:rgba(79,140,255,.08);border:1px solid rgba(79,140,255,.25);color:#6aa3ff;border-radius:4px;padding:4px 12px;font-size:12px;cursor:pointer;font-weight:600}
.ng-tbtn:hover{background:rgba(79,140,255,.2)}
.ng-tbtn-close{background:rgba(248,113,113,.08);border:1px solid rgba(248,113,113,.25);color:#f87171;border-radius:4px;padding:4px 12px;font-size:12px;cursor:pointer;font-weight:600}
.ng-tbtn-close:hover{background:rgba(248,113,113,.2)}
/* Icon-only variant — tighter padding so the topbar fits more
   buttons without wrapping. min-width keeps tap area readable. */
.ng-tbtn-icon{padding:4px 8px;min-width:30px;display:inline-flex;align-items:center;justify-content:center;font-size:14px;line-height:1}
.ng-tbtn-icon.ng-tbtn-close{padding:4px 8px}
.ng-topbar-divider{width:1px;height:18px;background:var(--ng-border2);margin:0 2px;flex-shrink:0}
.ng-main{display:flex;flex:1;overflow:hidden}
/* Sidebar */
.ng-sidebar{width:200px;background:var(--ng-input);border-right:1px solid var(--ng-border2);overflow-y:auto;flex-shrink:0;transition:width .15s ease}
.ng-sidebar.ng-collapsed{width:28px;cursor:pointer;overflow:hidden}
.ng-sidebar.ng-collapsed:hover{background:rgba(79,140,255,.08)}
.ng-sidebar.ng-collapsed .ng-sidebar-header-text,
.ng-sidebar.ng-collapsed .ng-sidebar-search,
.ng-sidebar.ng-collapsed .ng-cat{display:none}
.ng-sidebar.ng-collapsed .ng-sidebar-header{padding:8px 0;justify-content:center;border-bottom:none}
.ng-sidebar-header{padding:10px 12px;font-size:11px;font-weight:700;color:var(--ng-dim2);text-transform:uppercase;letter-spacing:.5px;border-bottom:1px solid var(--ng-border2);display:flex;align-items:center;justify-content:space-between;gap:6px}
.ng-sidebar-toggle{background:transparent;border:1px solid var(--ng-border2);color:var(--ng-dim2);width:18px;height:18px;border-radius:3px;font-size:9px;line-height:1;cursor:pointer;padding:0;flex-shrink:0}
.ng-sidebar-toggle:hover{color:#6aa3ff;border-color:#4f8cff}
.ng-sidebar-search{padding:8px 10px;border-bottom:1px solid var(--ng-border2)}
.ng-sidebar-search input{width:100%;background:var(--ng-bg);border:1px solid var(--ng-border2);color:var(--ng-text);border-radius:4px;padding:5px 8px;font-size:12px;outline:none}
.ng-sidebar-search input:focus{border-color:#4f8cff}
.ng-cat-header{padding:8px 12px;font-size:11px;font-weight:700;color:var(--ng-dim);cursor:pointer;display:flex;align-items:center;gap:5px;text-transform:uppercase;letter-spacing:.4px}
.ng-cat-header:hover{color:var(--ng-text)}
.ng-cat-arrow{font-size:7px;transition:transform .15s}
.ng-cat.ng-open .ng-cat-arrow{transform:rotate(90deg)}
.ng-cat-items{display:none;padding:0 0 6px}
.ng-cat.ng-open .ng-cat-items{display:block}
.ng-cat-item{padding:5px 12px 5px 24px;font-size:12px;color:var(--ng-dim);cursor:grab;display:flex;align-items:center;gap:6px;border-radius:4px;margin:0 6px}
.ng-cat-item:hover{background:var(--ng-surface);color:var(--ng-text)}
/* Canvas */
.ng-canvas-area{flex:1;position:relative;overflow:hidden;cursor:grab}
.ng-canvas-area.ng-panning{cursor:grabbing}
.ng-canvas{position:absolute;top:0;left:0;width:20000px;height:20000px;transform-origin:0 0;z-index:5}
.ng-grid-canvas,.ng-wire-canvas{position:absolute;top:0;left:0}
.ng-grid-canvas{z-index:0}
.ng-wire-canvas{pointer-events:none;z-index:2}
/* Nodes */
.ng-node{position:absolute;min-width:190px;max-width:420px;background:var(--ng-surface);border:2px solid var(--ng-border);border-radius:10px;box-shadow:0 2px 12px var(--ng-shadow);z-index:10;cursor:move;user-select:none}
.ng-node:not(.ng-coll){z-index:12}
.ng-node:hover{border-color:#3350a0;z-index:14}
.ng-node.ng-sel{border-color:#4f8cff;box-shadow:0 0 0 3px rgba(79,140,255,.25),0 0 24px rgba(79,140,255,.55),0 4px 20px var(--ng-shadow);z-index:16}
.ng-node.ng-connected{border-color:#fbbf24;box-shadow:0 0 0 2px rgba(251,191,36,.25),0 0 18px rgba(251,191,36,.45),0 2px 12px var(--ng-shadow)}
/* Brief flash applied on Ctrl+Click focus — helps the user spot the
   newly-centered node when the viewport jumps. Auto-removed after
   the keyframe finishes. */
@keyframes ng-focus-pulse {
  0%   { box-shadow: 0 0 0 4px rgba(167, 139, 250, 0.55), 0 0 32px rgba(167, 139, 250, 0.85), 0 4px 20px var(--ng-shadow); }
  60%  { box-shadow: 0 0 0 8px rgba(167, 139, 250, 0.18), 0 0 48px rgba(167, 139, 250, 0.40), 0 4px 20px var(--ng-shadow); }
  100% { box-shadow: 0 0 0 0 rgba(167, 139, 250, 0),    0 0 0 rgba(167, 139, 250, 0),    0 4px 20px var(--ng-shadow); }
}
.ng-node.ng-focus-pulse {
  animation: ng-focus-pulse 0.85s ease-out;
}
/* Header */
.ng-hdr{padding:9px 14px;font-size:18px;font-weight:800;text-transform:uppercase;letter-spacing:.5px;border-radius:8px 8px 0 0;display:flex;align-items:center;gap:7px;cursor:move}
.ng-hdr .ng-hi{font-size:20px}
.ng-hdr-name{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap;text-shadow:0 1px 2px rgba(0,0,0,.35)}
.ng-editbtn{font-size:11px;cursor:pointer;opacity:.45;padding:2px}
.ng-editbtn:hover{opacity:1;color:#fbbf24}
.ng-dupbtn{font-size:10px;cursor:pointer;opacity:.35;padding:2px;margin-left:4px}
.ng-dupbtn:hover{opacity:1;color:#4f8cff}
.ng-hdr-name + .ng-editbtn,.ng-hdr-name + .ng-dupbtn{margin-left:auto}
.ng-cbtn{font-size:9px;cursor:pointer;opacity:.5;padding:2px;margin-left:4px}
.ng-cbtn:hover{opacity:1}
.ng-attachbtn{font-size:11px;cursor:pointer;opacity:.45;padding:2px;margin-left:4px;line-height:1}
.ng-attachbtn:hover{opacity:1;color:#fbbf24}
.ng-attachbtn.ng-attached{opacity:1;color:#fbbf24;text-shadow:0 0 4px rgba(251,191,36,.6)}
.ng-attachbtn.ng-attaching{opacity:1;color:#fbbf24;animation:ng-attach-pulse 0.9s ease-in-out infinite}
@keyframes ng-attach-pulse{0%,100%{text-shadow:0 0 2px rgba(251,191,36,.4)}50%{text-shadow:0 0 10px rgba(251,191,36,1),0 0 18px rgba(251,191,36,.6)}}
/* While picking a target, hovering any node hints it's clickable */
.ng-canvas-area.ng-attaching .ng-node{cursor:crosshair}
.ng-canvas-area.ng-attaching .ng-node:hover{outline:2px dashed #fbbf24;outline-offset:2px}
/* Type colors */
.ng-t-t1 .ng-hdr{background:linear-gradient(135deg,rgba(79,140,255,.2),rgba(79,140,255,.08));color:#6aa3ff}
.ng-t-t2 .ng-hdr{background:linear-gradient(135deg,rgba(52,211,153,.2),rgba(52,211,153,.08));color:#34d399}
.ng-t-cost .ng-hdr{background:linear-gradient(135deg,rgba(251,191,36,.2),rgba(251,191,36,.08));color:#fbbf24}
.ng-t-sub .ng-hdr{background:linear-gradient(135deg,rgba(167,139,250,.2),rgba(167,139,250,.08));color:#a78bfa}
.ng-tt-po .ng-hdr{background:linear-gradient(135deg,rgba(34,211,238,.22),rgba(34,211,238,.08));color:#22d3ee}
.ng-tt-inv .ng-hdr{background:linear-gradient(135deg,rgba(252,231,243,.22),rgba(252,231,243,.08));color:#fce7f3}
.ng-t-co .ng-hdr{background:linear-gradient(135deg,rgba(220,38,38,.28),rgba(236,72,153,.16));color:#ec4899}
.ng-t-math .ng-hdr{background:linear-gradient(135deg,rgba(99,102,241,.2),rgba(99,102,241,.08));color:#818cf8}
.ng-t-watch .ng-hdr{background:linear-gradient(135deg,rgba(251,191,36,.15),rgba(251,191,36,.05));color:#fbbf24}
.ng-t-note .ng-hdr{background:linear-gradient(135deg,rgba(156,163,175,.15),rgba(156,163,175,.05));color:#9ca3af}
.ng-t-job .ng-hdr{background:linear-gradient(135deg,rgba(59,130,246,.25),rgba(59,130,246,.1));color:#60a5fa}
.ng-t-job{min-width:240px;border-color:#2563eb40}
.ng-t-wip .ng-hdr{background:linear-gradient(135deg,rgba(251,191,36,.2),rgba(236,72,153,.1));color:#fbbf24}
.ng-t-wip{min-width:340px;border-color:#f59e0b40}
.ng-wip-row{gap:10px}
.ng-wip-lbl{flex:1;color:var(--ng-dim);font-size:13px;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
.ng-wip-chip{display:inline-block;min-width:120px;text-align:right;font-family:'Courier New',monospace;font-size:15px;font-weight:700;color:#fbbf24;background:var(--ng-input);border:1px solid var(--ng-border2);border-radius:4px;padding:4px 10px;cursor:pointer;transition:all .12s;flex-shrink:0;user-select:none}
.ng-wip-chip:hover{border-color:#4f8cff;color:#fff;box-shadow:0 0 0 2px rgba(79,140,255,.15)}
.ng-wip-chip-input{width:120px;font-family:'Courier New',monospace;font-size:15px;font-weight:700;color:#fbbf24;background:var(--ng-input);border:1px solid #4f8cff;border-radius:4px;padding:4px 10px;outline:none;text-align:right;box-shadow:0 0 0 2px rgba(79,140,255,.2)}
/* Ports */
.ng-ports{padding:6px 0}
.ng-node.ng-coll .ng-ports{display:none}
.ng-pr{display:flex;align-items:center;padding:3px 0;min-height:24px}
.ng-p{width:13px;height:13px;border-radius:50%;border:2px solid #3a4570;background:var(--ng-port-bg);cursor:crosshair;flex-shrink:0;transition:all .1s;z-index:5}
.ng-p:hover{border-color:#4f8cff;background:#4f8cff;box-shadow:0 0 10px rgba(79,140,255,.6);transform:scale(1.4)}
.ng-p.ng-pc{background:#4f8cff;border-color:#4f8cff}
.ng-pi{margin-left:-7px;margin-right:8px}
.ng-po{margin-right:-7px;margin-left:8px}
.ng-p-currency{border-color:#34d399}.ng-p-currency.ng-pc{background:#34d399;border-color:#34d399}
.ng-p-percent{border-color:#fbbf24}.ng-p-percent.ng-pc{background:#fbbf24;border-color:#fbbf24}
.ng-p-number{border-color:#a78bfa}.ng-p-number.ng-pc{background:#a78bfa;border-color:#a78bfa}
.ng-pv-top,.ng-pv-bot{position:absolute;left:50%;transform:translateX(-50%);width:16px;height:16px;border-width:2px}
.ng-pv-top{top:-8px}
.ng-pv-bot{bottom:-8px}
.ng-pv-top:hover,.ng-pv-bot:hover{transform:translateX(-50%) scale(1.4)}
/* Port labels — input labels left-aligned, output labels right-aligned */
.ng-pl{font-size:13px;color:var(--ng-dim);flex:1}
.ng-pv{font-family:'Courier New',monospace;font-size:13px;color:#8899cc;font-weight:600;text-align:right}
/* Progress bar — click to edit */
.ng-progress-wrap{padding:6px 12px 4px;cursor:pointer}
.ng-progress-wrap:hover .ng-progress{box-shadow:0 0 0 1px rgba(79,140,255,.4)}
.ng-progress{height:9px;background:var(--ng-input);border-radius:5px;overflow:hidden;border:1px solid var(--ng-border2)}
.ng-progress-fill{height:100%;border-radius:4px;transition:width .3s}
.ng-progress-label{font-size:12px;color:var(--ng-dim2);text-align:center;padding:0 0 4px;display:flex;align-items:center;justify-content:center;gap:3px;cursor:pointer}
.ng-progress-label:hover .ng-pct-val{color:#fff;text-shadow:0 0 8px rgba(251,191,36,.6)}
.ng-pct-val{font-family:'Courier New',monospace;font-weight:700;color:#fbbf24;font-size:13px}
/* Watch value */
.ng-wv{padding:8px 14px;font-family:'Courier New',monospace;font-size:24px;font-weight:700;text-align:center;color:var(--ng-text);background:var(--ng-input);margin:4px 10px 8px;border-radius:6px;border:1px solid var(--ng-border2);min-width:110px}
.ng-wv.ng-vp{color:#34d399}
.ng-wv.ng-vn{color:#f87171}
/* Flashy Watch KPI — octopus tentacle displays */
.ng-watch-kpi{padding:22px 24px;font-family:'Courier New',monospace;font-size:40px;font-weight:800;text-align:center;color:#fbbf24;background:linear-gradient(135deg,#05080f 0%,#1a1230 50%,#0a1424 100%);margin:10px 14px 6px;border-radius:12px;border:2px solid rgba(251,191,36,.4);min-width:210px;letter-spacing:2px;text-shadow:0 0 18px rgba(251,191,36,.7),0 0 36px rgba(251,191,36,.35);box-shadow:inset 0 0 24px rgba(251,191,36,.08),0 0 22px rgba(251,191,36,.18),0 4px 16px rgba(0,0,0,.5);position:relative;overflow:hidden;animation:ng-kpi-pulse-amber 2.8s ease-in-out infinite}
.ng-watch-kpi::before{content:"";position:absolute;inset:0;background:linear-gradient(120deg,transparent 30%,rgba(255,255,255,.06) 50%,transparent 70%);animation:ng-kpi-shine 5s ease-in-out infinite;pointer-events:none;z-index:0}
.ng-watch-kpi-val{position:relative;z-index:1;display:block}
@keyframes ng-kpi-pulse-amber{0%,100%{box-shadow:inset 0 0 24px rgba(251,191,36,.08),0 0 22px rgba(251,191,36,.18),0 4px 16px rgba(0,0,0,.5)}50%{box-shadow:inset 0 0 32px rgba(251,191,36,.18),0 0 38px rgba(251,191,36,.42),0 4px 16px rgba(0,0,0,.5)}}
@keyframes ng-kpi-pulse-green{0%,100%{box-shadow:inset 0 0 24px rgba(52,211,153,.08),0 0 22px rgba(52,211,153,.18),0 4px 16px rgba(0,0,0,.5)}50%{box-shadow:inset 0 0 32px rgba(52,211,153,.2),0 0 38px rgba(52,211,153,.48),0 4px 16px rgba(0,0,0,.5)}}
@keyframes ng-kpi-pulse-red{0%,100%{box-shadow:inset 0 0 24px rgba(248,113,113,.08),0 0 22px rgba(248,113,113,.18),0 4px 16px rgba(0,0,0,.5)}50%{box-shadow:inset 0 0 32px rgba(248,113,113,.2),0 0 38px rgba(248,113,113,.48),0 4px 16px rgba(0,0,0,.5)}}
@keyframes ng-kpi-shine{0%,100%{transform:translateX(-100%)}50%{transform:translateX(100%)}}
.ng-watch-kpi.ng-vp{color:#34d399;border-color:rgba(52,211,153,.5);text-shadow:0 0 18px rgba(52,211,153,.7),0 0 36px rgba(52,211,153,.35);animation:ng-kpi-pulse-green 2.8s ease-in-out infinite;background:linear-gradient(135deg,#05080f 0%,#0a2418 50%,#0a1424 100%)}
.ng-watch-kpi.ng-vn{color:#f87171;border-color:rgba(248,113,113,.5);text-shadow:0 0 18px rgba(248,113,113,.7),0 0 36px rgba(248,113,113,.35);animation:ng-kpi-pulse-red 2.8s ease-in-out infinite;background:linear-gradient(135deg,#05080f 0%,#2a0e14 50%,#0a1424 100%)}
.ng-t-watch{min-width:260px;border-color:rgba(251,191,36,.3)}
.ng-t-watch .ng-hdr{background:linear-gradient(135deg,rgba(251,191,36,.25),rgba(236,72,153,.15));color:#fbbf24;text-shadow:0 0 8px rgba(251,191,36,.4);font-size:22px;padding:12px 16px}
.ng-t-watch .ng-hdr .ng-hi{font-size:26px}
.ng-t-watch .ng-hdr-name{font-size:22px;letter-spacing:1px}
.ng-t-watch .ng-p.ng-pi{position:absolute;left:-7px;top:50%;transform:translateY(-50%)}
.ng-wv-label{font-size:11px;color:#fbbf24;text-align:center;padding:0 12px 10px;text-transform:uppercase;letter-spacing:1px;font-weight:700;opacity:.85}
/* Editable input */
.ng-edit-val{padding:6px 10px 8px}
.ng-edit-val label{font-size:10px;color:var(--ng-dim2);display:block;text-align:center;margin-bottom:3px}
.ng-edit-val input{width:100%;background:var(--ng-input);border:1px solid var(--ng-border2);color:var(--ng-text);font-family:'Courier New',monospace;font-size:15px;font-weight:700;padding:5px 8px;border-radius:5px;text-align:center;outline:none}
.ng-edit-val input:focus{border-color:#4f8cff;box-shadow:0 0 0 2px rgba(79,140,255,.15)}
.ng-node input[type="number"]::-webkit-inner-spin-button,.ng-node input[type="number"]::-webkit-outer-spin-button{-webkit-appearance:none;margin:0}
.ng-node input[type="number"]{-moz-appearance:textfield}
/* Cost sub-items */
.ng-subitems{padding:4px 12px 6px;max-height:220px;overflow-y:auto}
.ng-node.ng-coll .ng-subitems{display:none}
.ng-subitem{display:flex;align-items:center;gap:5px;padding:3px 0;font-size:11px;border-bottom:1px solid var(--ng-border2)}
.ng-subitem:last-child{border-bottom:none}
.ng-si-hdr{display:flex;gap:5px;padding:0 0 3px;font-size:9px;color:var(--ng-dim2);text-transform:uppercase;letter-spacing:.3px;border-bottom:1px solid var(--ng-border2);margin-bottom:3px}
.ng-si-hdr span{flex:1;min-width:0}
.ng-si-hdr .hd{flex:none}
.ng-si-hdr .hd-date{width:105px}
.ng-si-hdr .hd-sm{width:55px;text-align:right}
.ng-si-hdr .hd-flex{flex:1;min-width:0}
.ng-si-hdr .hd-del{width:17px}
.ng-si-f{background:var(--ng-input);border:1px solid var(--ng-border2);color:var(--ng-dim);font-size:11px;padding:3px 5px;border-radius:3px;outline:none;font-family:inherit;min-width:0}
.ng-si-f:focus{border-color:#4f8cff}
.ng-si-date{width:105px;flex-shrink:0}
.ng-si-sm{width:55px;flex-shrink:0;text-align:right;font-family:'Courier New',monospace;color:#8899cc;font-weight:600}
.ng-si-sel{width:48px;flex-shrink:0;font-size:10px;padding:3px 3px;cursor:pointer}
.ng-si-val{font-family:'Courier New',monospace;font-size:11px;color:#fbbf24;font-weight:600;text-align:right;min-width:55px;flex-shrink:0}
.ng-subitem-del{color:#f87171;cursor:pointer;font-size:9px;opacity:.5;padding:0 3px;flex-shrink:0}
.ng-subitem-del:hover{opacity:1}
.ng-add-sub{font-size:10px;color:#4f8cff;cursor:pointer;padding:4px 0;text-align:center}
.ng-add-sub:hover{color:#6aa3ff;text-decoration:underline}
.ng-sub-total{font-family:'Courier New',monospace;font-size:15px;font-weight:700;color:#fbbf24;text-align:center;padding:5px 12px 6px;border-top:1px solid var(--ng-border2)}
/* Collapsed row */
.ng-coll-row{display:none}
.ng-node.ng-coll .ng-coll-row{display:flex;flex-direction:column;align-items:stretch;padding:4px 14px 8px;gap:4px}
.ng-node.ng-coll .ng-ports{display:none}
.ng-node.ng-coll .ng-edit-val{display:none}
.ng-node.ng-coll .ng-progress-wrap{display:none}
.ng-node.ng-coll .ng-progress{display:none}
.ng-node.ng-coll .ng-progress-label{display:none}
.ng-node.ng-coll .ng-subitems{display:none}
.ng-node.ng-coll .ng-wv{display:none}
.ng-node.ng-coll .ng-watch-kpi{display:none}
/* Collapsed progress bar */
.ng-coll-prog{height:5px;background:var(--ng-input);border-radius:3px;overflow:hidden;width:100%}
.ng-coll-prog-fill{height:100%;border-radius:3px;transition:width .3s}
.ng-coll-info{display:flex;justify-content:space-between;align-items:center}
.ng-coll-inline{display:flex;align-items:center;justify-content:center;gap:12px;font-family:'Courier New',monospace;font-size:13px;font-weight:700;padding:2px 0}
.ng-coll-inline .ng-cv-bud{font-size:13px}
.ng-coll-inline .ng-coll-val{font-size:13px}
.ng-coll-inline .ng-coll-pct{font-size:13px}
.ng-coll-sep{color:#4a5068;font-weight:400;font-size:13px}
.ng-coll-pct{font-family:'Courier New',monospace;font-size:11px;font-weight:700;color:#fbbf24}
/* Collapsed ports sit on the node edges, centered to full node height */
.ng-node.ng-coll .ng-coll-pi{position:absolute;left:-5px;top:50%;transform:translateY(-50%);width:10px;height:10px;border-radius:50%;background:#4f8cff;border:2px solid #4f8cff;cursor:crosshair;z-index:5}
.ng-node.ng-coll .ng-coll-po{position:absolute;right:-5px;top:50%;transform:translateY(-50%);width:10px;height:10px;border-radius:50%;background:#4f8cff;border:2px solid #4f8cff;cursor:crosshair;z-index:5}
.ng-coll-val{font-family:'Courier New',monospace;font-size:14px;font-weight:700;color:#8899cc;text-align:right}
.ng-coll-detail{display:flex;justify-content:space-between;align-items:center;gap:6px}
.ng-coll-lbl{font-size:9px;color:#6a7090;text-transform:uppercase;letter-spacing:.3px;white-space:nowrap}
.ng-coll-detail .ng-coll-val{font-size:12px}
.ng-cv-bud{color:#a78bfa}
.ng-cv-grn{color:#34d399}
.ng-cv-yel{color:#fbbf24}
.ng-cv-blu{color:#4f8cff}
/* Collapsed T1/T2 multi-row KPI — progress spans full width, KVs in 2-col grid */
.ng-node.ng-coll .ng-coll-t12{display:grid;grid-template-columns:1fr 1fr;column-gap:10px;row-gap:2px;width:100%;padding:2px 4px}
.ng-node.ng-coll .ng-coll-t12 .ng-coll-prog{grid-column:1 / -1}
.ng-coll-pctrow{display:flex;justify-content:center;padding:0 0 2px;grid-column:1 / -1}
.ng-coll-kv{display:flex;justify-content:space-between;align-items:center;gap:6px;padding:1px 0;min-width:0}
.ng-coll-kv .ng-coll-val{font-size:11px;text-align:right;flex:0 0 auto;overflow:hidden;text-overflow:ellipsis}
.ng-coll-kv .ng-coll-lbl{font-size:9px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}
/* Delete dialog */
.ng-del-overlay{position:fixed;inset:0;background:rgba(0,0,0,.55);z-index:9999;display:flex;align-items:center;justify-content:center}
.ng-del-box{background:#1a1d27;border:1px solid #2e3346;border-radius:10px;min-width:340px;max-width:420px;padding:24px;box-shadow:0 8px 32px rgba(0,0,0,.5)}
.ng-del-title{font-size:16px;font-weight:700;color:#e4e6f0;margin-bottom:6px}
.ng-del-sub{font-size:12px;color:#8b90a5;margin-bottom:20px}
.ng-del-btns{display:flex;flex-direction:column;gap:8px}
.ng-del-btn{padding:10px 16px;border:none;border-radius:6px;font-size:13px;font-weight:600;cursor:pointer;transition:background .15s}
.ng-del-remove{background:#2a4a8a;color:#e4e6f0}
.ng-del-remove:hover{background:#3b5fad}
.ng-del-permanent{background:#7f1d1d;color:#fca5a5}
.ng-del-permanent:hover{background:#991b1b}
.ng-del-cancel{background:#242836;color:#8b90a5;border:1px solid #2e3346}
.ng-del-cancel:hover{background:#2e3346}
/* Note */
.ng-note-body{padding:8px 12px 10px}
.ng-node.ng-coll .ng-note-body{display:none}
.ng-note-body textarea{width:100%;min-height:55px;background:var(--ng-input);border:1px solid var(--ng-border2);color:var(--ng-text);font-size:12px;padding:6px 8px;border-radius:4px;resize:vertical;outline:none;font-family:inherit;line-height:1.4}
.ng-note-body textarea:focus{border-color:#4f8cff}
/* Data picker overlay */
.ng-picker-overlay{position:absolute;inset:0;background:rgba(0,0,0,.55);z-index:50;display:flex;align-items:center;justify-content:center}
.ng-picker-box{background:var(--ng-surface);border:1px solid var(--ng-border);border-radius:10px;min-width:320px;max-width:460px;max-height:70vh;display:flex;flex-direction:column;box-shadow:0 8px 32px rgba(0,0,0,.5)}
.ng-picker-title{padding:14px 18px;font-size:14px;font-weight:700;color:var(--ng-text);border-bottom:1px solid var(--ng-border2)}
.ng-picker-list{overflow-y:auto;padding:6px 0}
.ng-picker-item{padding:10px 18px;font-size:13px;color:var(--ng-dim);cursor:pointer;border-bottom:1px solid var(--ng-border2)}
.ng-picker-item:last-child{border-bottom:none}
.ng-picker-item:hover{background:rgba(79,140,255,.1);color:var(--ng-text)}
.ng-picker-new{color:#4f8cff;font-weight:600}
.ng-picker-new:hover{background:rgba(79,140,255,.15)}
.ng-picker-loaded{opacity:.55;font-style:italic}
.ng-picker-loaded:hover{opacity:.85;background:rgba(52,211,153,.08)}
/* HUD */
.ng-zoom{position:absolute;bottom:10px;left:10px;font-size:11px;color:var(--ng-dim2);background:var(--ng-surface);border:1px solid var(--ng-border2);border-radius:4px;padding:4px 10px;z-index:10}
.ng-help{position:absolute;bottom:10px;right:10px;font-size:10px;color:var(--ng-dim2);background:var(--ng-surface);border:1px solid var(--ng-border2);border-radius:4px;padding:5px 12px;z-index:10;line-height:1.5}

/* ── Outlook-style ribbon toolbar ───────────────────────── */
/* Activates when .ng-topbar has the .ng-ribbon modifier class.
   Each .ng-ribbon-btn is a vertical button (colored emoji on top,
   label below); buttons cluster into named .ng-ribbon-group blocks
   with a small caption underneath each group; vertical .ng-ribbon-sep
   lines divide groups. Modeled on the Outlook ribbon screenshot
   the user shared. */
.ng-topbar.ng-ribbon{
  align-items:stretch;
  gap:0;
  padding:2px 10px 0;
  flex-wrap:wrap;
  border-bottom:1px solid var(--ng-border2);
}
.ng-topbar.ng-ribbon .ng-ribbon-title{
  display:flex;
  flex-direction:column;
  justify-content:center;
  gap:1px;
  padding:2px 12px 4px 0;
  border-right:1px solid var(--ng-border2);
  margin-right:4px;
}
/* Row 1 of the title block — "⚡ Node Graph" + Save indicator
   inline, separated by a small gap. */
.ng-topbar.ng-ribbon .ng-ribbon-title-row{
  display:flex;
  align-items:center;
  gap:8px;
}
.ng-topbar.ng-ribbon .ng-topbar-title{
  font-size:13px;
  line-height:1.1;
  white-space:nowrap;
}
/* Row 2 — subtitle restored. */
.ng-topbar.ng-ribbon .ng-topbar-sub{
  display:block;
  font-size:10px;
  color:var(--ng-dim2,#6b7280);
  white-space:nowrap;
}
.ng-topbar.ng-ribbon .ng-save-indicator{
  font-size:10px;
  padding:2px 8px;
  margin-top:0;
  /* Removed align-self:flex-start so it sits centered with the
     title text on its own row instead of stacking below it. */
}
/* Spacer enabled — flex-grow:1 to consume slack space and push
   button groups to the right edge of the ribbon. The title block
   stays on the left. */
.ng-topbar.ng-ribbon .ng-topbar-spacer{
  display:block;
  flex:1 1 auto;
  min-width:8px;
}
.ng-topbar.ng-ribbon .ng-ribbon-group{
  display:flex;
  flex-direction:column;
  align-items:center;
  flex:0 0 auto;
  padding:0 3px;
}
.ng-topbar.ng-ribbon .ng-ribbon-group-buttons{
  display:flex;
  align-items:stretch;
  gap:0;
}
.ng-topbar.ng-ribbon .ng-ribbon-group-name{
  font-size:9px;
  text-transform:uppercase;
  letter-spacing:0.5px;
  color:var(--ng-dim2,#6b7280);
  font-weight:500;
  margin-top:0;
  /* Padding-bottom 0 + line-height 1 so the caption text sits flush
     against the bottom border with no dead space below. */
  padding:2px 4px 0;
  line-height:1;
  border-top:1px solid var(--ng-border2);
  width:100%;
  text-align:center;
  /* Pull the caption down to the topbar\'s bottom edge so the row
     ends right where the caption text ends. */
  margin-bottom:-1px;
}
.ng-topbar.ng-ribbon .ng-ribbon-sep{
  flex:0 0 1px;
  width:1px;
  background:var(--ng-border2);
  margin:2px 3px 0;
}
.ng-topbar.ng-ribbon .ng-ribbon-btn{
  display:flex;
  flex-direction:column;
  align-items:center;
  justify-content:center;
  gap:1px;
  min-width:54px;
  padding:2px 8px 3px;
  background:transparent;
  border:1px solid transparent;
  border-radius:5px;
  color:var(--ng-text);
  font-family:inherit;
  font-size:10px;
  font-weight:500;
  cursor:pointer;
  transition:background .12s,border-color .12s,transform .06s;
}
.ng-topbar.ng-ribbon .ng-ribbon-icon{
  font-size:18px;
  line-height:1;
  display:inline-block;
  filter:saturate(1.05);
  transition:transform .12s;
}
.ng-topbar.ng-ribbon .ng-ribbon-label{
  font-size:10px;
  font-weight:500;
  color:var(--ng-dim,#b8becc);
  white-space:nowrap;
  line-height:1.1;
}
.ng-topbar.ng-ribbon .ng-ribbon-btn:hover{
  background:rgba(255,255,255,0.05);
  border-color:rgba(255,255,255,0.08);
}
.ng-topbar.ng-ribbon .ng-ribbon-btn:hover .ng-ribbon-label{
  color:var(--ng-text,#fff);
}
.ng-topbar.ng-ribbon .ng-ribbon-btn:hover .ng-ribbon-icon{
  transform:translateY(-1px);
}
/* AI button gets a gradient-tint background so it reads as the
   "primary action" without the user having to hover. */
.ng-topbar.ng-ribbon .ng-ribbon-btn-ai{
  background:linear-gradient(135deg,rgba(139,92,246,.18),rgba(79,140,255,.18));
  border-color:rgba(139,92,246,.35);
}
.ng-topbar.ng-ribbon .ng-ribbon-btn-ai:hover{
  background:linear-gradient(135deg,rgba(139,92,246,.28),rgba(79,140,255,.28));
  border-color:rgba(139,92,246,.55);
}

/* Light-mode parity for the ribbon dividers + hover */
body.light-mode .ng-topbar.ng-ribbon .ng-ribbon-group-name{
  border-top-color:var(--ng-border2,#d4d8e0);
  color:#6b7280;
}
body.light-mode .ng-topbar.ng-ribbon .ng-ribbon-sep,
body.light-mode .ng-topbar.ng-ribbon .ng-ribbon-title{
  background:var(--ng-border2,#d4d8e0);
}
body.light-mode .ng-topbar.ng-ribbon .ng-ribbon-btn:hover{
  background:rgba(0,0,0,.04);
  border-color:rgba(0,0,0,.06);
}
