@shaztech/video-pipeline 1.0.4 → 1.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- .react-flow{direction:ltr;--xy-edge-stroke-default: #b1b1b7;--xy-edge-stroke-width-default: 1;--xy-edge-stroke-selected-default: #555;--xy-connectionline-stroke-default: #b1b1b7;--xy-connectionline-stroke-width-default: 1;--xy-attribution-background-color-default: rgba(255, 255, 255, .5);--xy-minimap-background-color-default: #fff;--xy-minimap-mask-background-color-default: rgba(240, 240, 240, .6);--xy-minimap-mask-stroke-color-default: transparent;--xy-minimap-mask-stroke-width-default: 1;--xy-minimap-node-background-color-default: #e2e2e2;--xy-minimap-node-stroke-color-default: transparent;--xy-minimap-node-stroke-width-default: 2;--xy-background-color-default: transparent;--xy-background-pattern-dots-color-default: #91919a;--xy-background-pattern-lines-color-default: #eee;--xy-background-pattern-cross-color-default: #e2e2e2;background-color:var(--xy-background-color, var(--xy-background-color-default));--xy-node-color-default: inherit;--xy-node-border-default: 1px solid #1a192b;--xy-node-background-color-default: #fff;--xy-node-group-background-color-default: rgba(240, 240, 240, .25);--xy-node-boxshadow-hover-default: 0 1px 4px 1px rgba(0, 0, 0, .08);--xy-node-boxshadow-selected-default: 0 0 0 .5px #1a192b;--xy-node-border-radius-default: 3px;--xy-handle-background-color-default: #1a192b;--xy-handle-border-color-default: #fff;--xy-selection-background-color-default: rgba(0, 89, 220, .08);--xy-selection-border-default: 1px dotted rgba(0, 89, 220, .8);--xy-controls-button-background-color-default: #fefefe;--xy-controls-button-background-color-hover-default: #f4f4f4;--xy-controls-button-color-default: inherit;--xy-controls-button-color-hover-default: inherit;--xy-controls-button-border-color-default: #eee;--xy-controls-box-shadow-default: 0 0 2px 1px rgba(0, 0, 0, .08);--xy-edge-label-background-color-default: #ffffff;--xy-edge-label-color-default: inherit;--xy-resize-background-color-default: #3367d9}.react-flow.dark{--xy-edge-stroke-default: #3e3e3e;--xy-edge-stroke-width-default: 1;--xy-edge-stroke-selected-default: #727272;--xy-connectionline-stroke-default: #b1b1b7;--xy-connectionline-stroke-width-default: 1;--xy-attribution-background-color-default: rgba(150, 150, 150, .25);--xy-minimap-background-color-default: #141414;--xy-minimap-mask-background-color-default: rgba(60, 60, 60, .6);--xy-minimap-mask-stroke-color-default: transparent;--xy-minimap-mask-stroke-width-default: 1;--xy-minimap-node-background-color-default: #2b2b2b;--xy-minimap-node-stroke-color-default: transparent;--xy-minimap-node-stroke-width-default: 2;--xy-background-color-default: #141414;--xy-background-pattern-dots-color-default: #777;--xy-background-pattern-lines-color-default: #777;--xy-background-pattern-cross-color-default: #777;--xy-node-color-default: #f8f8f8;--xy-node-border-default: 1px solid #3c3c3c;--xy-node-background-color-default: #1e1e1e;--xy-node-group-background-color-default: rgba(240, 240, 240, .25);--xy-node-boxshadow-hover-default: 0 1px 4px 1px rgba(255, 255, 255, .08);--xy-node-boxshadow-selected-default: 0 0 0 .5px #999;--xy-handle-background-color-default: #bebebe;--xy-handle-border-color-default: #1e1e1e;--xy-selection-background-color-default: rgba(200, 200, 220, .08);--xy-selection-border-default: 1px dotted rgba(200, 200, 220, .8);--xy-controls-button-background-color-default: #2b2b2b;--xy-controls-button-background-color-hover-default: #3e3e3e;--xy-controls-button-color-default: #f8f8f8;--xy-controls-button-color-hover-default: #fff;--xy-controls-button-border-color-default: #5b5b5b;--xy-controls-box-shadow-default: 0 0 2px 1px rgba(0, 0, 0, .08);--xy-edge-label-background-color-default: #141414;--xy-edge-label-color-default: #f8f8f8}.react-flow__background{background-color:var(--xy-background-color-props, var(--xy-background-color, var(--xy-background-color-default)));pointer-events:none;z-index:-1}.react-flow__container{position:absolute;width:100%;height:100%;top:0;left:0}.react-flow__pane{z-index:1}.react-flow__pane.draggable{cursor:grab}.react-flow__pane.dragging{cursor:grabbing}.react-flow__pane.selection{cursor:pointer}.react-flow__viewport{transform-origin:0 0;z-index:2;pointer-events:none}.react-flow__renderer{z-index:4}.react-flow__selection{z-index:6}.react-flow__nodesselection-rect:focus,.react-flow__nodesselection-rect:focus-visible{outline:none}.react-flow__edge-path{stroke:var(--xy-edge-stroke, var(--xy-edge-stroke-default));stroke-width:var(--xy-edge-stroke-width, var(--xy-edge-stroke-width-default));fill:none}.react-flow__connection-path{stroke:var(--xy-connectionline-stroke, var(--xy-connectionline-stroke-default));stroke-width:var(--xy-connectionline-stroke-width, var(--xy-connectionline-stroke-width-default));fill:none}.react-flow .react-flow__edges{position:absolute}.react-flow .react-flow__edges svg{overflow:visible;position:absolute;pointer-events:none}.react-flow__edge{pointer-events:visibleStroke}.react-flow__edge.selectable{cursor:pointer}.react-flow__edge.animated path{stroke-dasharray:5;animation:dashdraw .5s linear infinite}.react-flow__edge.animated path.react-flow__edge-interaction{stroke-dasharray:none;animation:none}.react-flow__edge.inactive{pointer-events:none}.react-flow__edge.selected,.react-flow__edge:focus,.react-flow__edge:focus-visible{outline:none}.react-flow__edge.selected .react-flow__edge-path,.react-flow__edge.selectable:focus .react-flow__edge-path,.react-flow__edge.selectable:focus-visible .react-flow__edge-path{stroke:var(--xy-edge-stroke-selected, var(--xy-edge-stroke-selected-default))}.react-flow__edge-textwrapper{pointer-events:all}.react-flow__edge .react-flow__edge-text{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none}.react-flow__arrowhead polyline{stroke:var(--xy-edge-stroke, var(--xy-edge-stroke-default))}.react-flow__arrowhead polyline.arrowclosed{fill:var(--xy-edge-stroke, var(--xy-edge-stroke-default))}.react-flow__connection{pointer-events:none}.react-flow__connection .animated{stroke-dasharray:5;animation:dashdraw .5s linear infinite}svg.react-flow__connectionline{z-index:1001;overflow:visible;position:absolute}.react-flow__nodes{pointer-events:none;transform-origin:0 0}.react-flow__node{position:absolute;-webkit-user-select:none;-moz-user-select:none;user-select:none;pointer-events:all;transform-origin:0 0;box-sizing:border-box;cursor:default}.react-flow__node.selectable{cursor:pointer}.react-flow__node.draggable{cursor:grab;pointer-events:all}.react-flow__node.draggable.dragging{cursor:grabbing}.react-flow__nodesselection{z-index:3;transform-origin:left top;pointer-events:none}.react-flow__nodesselection-rect{position:absolute;pointer-events:all;cursor:grab}.react-flow__handle{position:absolute;pointer-events:none;min-width:5px;min-height:5px;width:6px;height:6px;background-color:var(--xy-handle-background-color, var(--xy-handle-background-color-default));border:1px solid var(--xy-handle-border-color, var(--xy-handle-border-color-default));border-radius:100%}.react-flow__handle.connectingfrom{pointer-events:all}.react-flow__handle.connectionindicator{pointer-events:all;cursor:crosshair}.react-flow__handle-bottom{top:auto;left:50%;bottom:0;transform:translate(-50%,50%)}.react-flow__handle-top{top:0;left:50%;transform:translate(-50%,-50%)}.react-flow__handle-left{top:50%;left:0;transform:translate(-50%,-50%)}.react-flow__handle-right{top:50%;right:0;transform:translate(50%,-50%)}.react-flow__edgeupdater{cursor:move;pointer-events:all}.react-flow__pane.selection .react-flow__panel{pointer-events:none}.react-flow__panel{position:absolute;z-index:5;margin:15px}.react-flow__panel.top{top:0}.react-flow__panel.bottom{bottom:0}.react-flow__panel.top.center,.react-flow__panel.bottom.center{left:50%;transform:translate(-15px) translate(-50%)}.react-flow__panel.left{left:0}.react-flow__panel.right{right:0}.react-flow__panel.left.center,.react-flow__panel.right.center{top:50%;transform:translateY(-15px) translateY(-50%)}.react-flow__attribution{font-size:10px;background:var(--xy-attribution-background-color, var(--xy-attribution-background-color-default));padding:2px 3px;margin:0}.react-flow__attribution a{text-decoration:none;color:#999}@keyframes dashdraw{0%{stroke-dashoffset:10}}.react-flow__edgelabel-renderer{position:absolute;width:100%;height:100%;pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none;left:0;top:0}.react-flow__viewport-portal{position:absolute;width:100%;height:100%;left:0;top:0;-webkit-user-select:none;-moz-user-select:none;user-select:none}.react-flow__minimap{background:var( --xy-minimap-background-color-props, var(--xy-minimap-background-color, var(--xy-minimap-background-color-default)) )}.react-flow__minimap-svg{display:block}.react-flow__minimap-mask{fill:var( --xy-minimap-mask-background-color-props, var(--xy-minimap-mask-background-color, var(--xy-minimap-mask-background-color-default)) );stroke:var( --xy-minimap-mask-stroke-color-props, var(--xy-minimap-mask-stroke-color, var(--xy-minimap-mask-stroke-color-default)) );stroke-width:var( --xy-minimap-mask-stroke-width-props, var(--xy-minimap-mask-stroke-width, var(--xy-minimap-mask-stroke-width-default)) )}.react-flow__minimap-node{fill:var( --xy-minimap-node-background-color-props, var(--xy-minimap-node-background-color, var(--xy-minimap-node-background-color-default)) );stroke:var( --xy-minimap-node-stroke-color-props, var(--xy-minimap-node-stroke-color, var(--xy-minimap-node-stroke-color-default)) );stroke-width:var( --xy-minimap-node-stroke-width-props, var(--xy-minimap-node-stroke-width, var(--xy-minimap-node-stroke-width-default)) )}.react-flow__background-pattern.dots{fill:var( --xy-background-pattern-color-props, var(--xy-background-pattern-color, var(--xy-background-pattern-dots-color-default)) )}.react-flow__background-pattern.lines{stroke:var( --xy-background-pattern-color-props, var(--xy-background-pattern-color, var(--xy-background-pattern-lines-color-default)) )}.react-flow__background-pattern.cross{stroke:var( --xy-background-pattern-color-props, var(--xy-background-pattern-color, var(--xy-background-pattern-cross-color-default)) )}.react-flow__controls{display:flex;flex-direction:column;box-shadow:var(--xy-controls-box-shadow, var(--xy-controls-box-shadow-default))}.react-flow__controls.horizontal{flex-direction:row}.react-flow__controls-button{display:flex;justify-content:center;align-items:center;height:26px;width:26px;padding:4px;border:none;background:var(--xy-controls-button-background-color, var(--xy-controls-button-background-color-default));border-bottom:1px solid var( --xy-controls-button-border-color-props, var(--xy-controls-button-border-color, var(--xy-controls-button-border-color-default)) );color:var( --xy-controls-button-color-props, var(--xy-controls-button-color, var(--xy-controls-button-color-default)) );cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none}.react-flow__controls-button svg{width:100%;max-width:12px;max-height:12px;fill:currentColor}.react-flow__edge.updating .react-flow__edge-path{stroke:#777}.react-flow__edge-text{font-size:10px}.react-flow__node.selectable:focus,.react-flow__node.selectable:focus-visible{outline:none}.react-flow__node-input,.react-flow__node-default,.react-flow__node-output,.react-flow__node-group{padding:10px;border-radius:var(--xy-node-border-radius, var(--xy-node-border-radius-default));width:150px;font-size:12px;color:var(--xy-node-color, var(--xy-node-color-default));text-align:center;border:var(--xy-node-border, var(--xy-node-border-default));background-color:var(--xy-node-background-color, var(--xy-node-background-color-default))}.react-flow__node-input.selectable:hover,.react-flow__node-default.selectable:hover,.react-flow__node-output.selectable:hover,.react-flow__node-group.selectable:hover{box-shadow:var(--xy-node-boxshadow-hover, var(--xy-node-boxshadow-hover-default))}.react-flow__node-input.selectable.selected,.react-flow__node-input.selectable:focus,.react-flow__node-input.selectable:focus-visible,.react-flow__node-default.selectable.selected,.react-flow__node-default.selectable:focus,.react-flow__node-default.selectable:focus-visible,.react-flow__node-output.selectable.selected,.react-flow__node-output.selectable:focus,.react-flow__node-output.selectable:focus-visible,.react-flow__node-group.selectable.selected,.react-flow__node-group.selectable:focus,.react-flow__node-group.selectable:focus-visible{box-shadow:var(--xy-node-boxshadow-selected, var(--xy-node-boxshadow-selected-default))}.react-flow__node-group{background-color:var(--xy-node-group-background-color, var(--xy-node-group-background-color-default))}.react-flow__nodesselection-rect,.react-flow__selection{background:var(--xy-selection-background-color, var(--xy-selection-background-color-default));border:var(--xy-selection-border, var(--xy-selection-border-default))}.react-flow__nodesselection-rect:focus,.react-flow__nodesselection-rect:focus-visible,.react-flow__selection:focus,.react-flow__selection:focus-visible{outline:none}.react-flow__controls-button:hover{background:var( --xy-controls-button-background-color-hover-props, var(--xy-controls-button-background-color-hover, var(--xy-controls-button-background-color-hover-default)) );color:var( --xy-controls-button-color-hover-props, var(--xy-controls-button-color-hover, var(--xy-controls-button-color-hover-default)) )}.react-flow__controls-button:disabled{pointer-events:none}.react-flow__controls-button:disabled svg{fill-opacity:.4}.react-flow__controls-button:last-child{border-bottom:none}.react-flow__controls.horizontal .react-flow__controls-button{border-bottom:none;border-right:1px solid var( --xy-controls-button-border-color-props, var(--xy-controls-button-border-color, var(--xy-controls-button-border-color-default)) )}.react-flow__controls.horizontal .react-flow__controls-button:last-child{border-right:none}.react-flow__resize-control{position:absolute}.react-flow__resize-control.left,.react-flow__resize-control.right{cursor:ew-resize}.react-flow__resize-control.top,.react-flow__resize-control.bottom{cursor:ns-resize}.react-flow__resize-control.top.left,.react-flow__resize-control.bottom.right{cursor:nwse-resize}.react-flow__resize-control.bottom.left,.react-flow__resize-control.top.right{cursor:nesw-resize}.react-flow__resize-control.handle{width:5px;height:5px;border:1px solid #fff;border-radius:1px;background-color:var(--xy-resize-background-color, var(--xy-resize-background-color-default));translate:-50% -50%}.react-flow__resize-control.handle.left{left:0;top:50%}.react-flow__resize-control.handle.right{left:100%;top:50%}.react-flow__resize-control.handle.top{left:50%;top:0}.react-flow__resize-control.handle.bottom{left:50%;top:100%}.react-flow__resize-control.handle.top.left,.react-flow__resize-control.handle.bottom.left{left:0}.react-flow__resize-control.handle.top.right,.react-flow__resize-control.handle.bottom.right{left:100%}.react-flow__resize-control.line{border-color:var(--xy-resize-background-color, var(--xy-resize-background-color-default));border-width:0;border-style:solid}.react-flow__resize-control.line.left,.react-flow__resize-control.line.right{width:1px;transform:translate(-50%);top:0;height:100%}.react-flow__resize-control.line.left{left:0;border-left-width:1px}.react-flow__resize-control.line.right{left:100%;border-right-width:1px}.react-flow__resize-control.line.top,.react-flow__resize-control.line.bottom{height:1px;transform:translateY(-50%);left:0;width:100%}.react-flow__resize-control.line.top{top:0;border-top-width:1px}.react-flow__resize-control.line.bottom{border-bottom-width:1px;top:100%}.react-flow__edge-textbg{fill:var(--xy-edge-label-background-color, var(--xy-edge-label-background-color-default))}.react-flow__edge-text{fill:var(--xy-edge-label-color, var(--xy-edge-label-color-default))}._toolbar_1uqg7_1{position:absolute;top:0;left:0;right:0;z-index:10;display:flex;align-items:center;justify-content:space-between;padding:8px 14px;background:#0d1117eb;-webkit-backdrop-filter:blur(8px);backdrop-filter:blur(8px);border-bottom:1px solid #21262d}._left_1uqg7_16{display:flex;align-items:center;gap:10px}._right_1uqg7_22{display:flex;align-items:center;gap:8px}._pipelineName_1uqg7_28{font-size:13px;font-weight:700;color:#e0e0e0;letter-spacing:.02em;cursor:text}._pipelineName_1uqg7_28:hover{color:#fff}._pipelineNameInput_1uqg7_40{font-size:13px;font-weight:700;color:#e0e0e0;letter-spacing:.02em;background:transparent;border:none;border-bottom:1px solid #388bfd;outline:none;padding:0;min-width:80px;width:auto}._divider_1uqg7_54{width:1px;height:20px;background:#21262d}._addBtn_1uqg7_60{border:none;border-radius:6px;font-size:12px;font-weight:600;padding:6px 12px;cursor:pointer;transition:opacity .15s,transform .1s}._addBtn_1uqg7_60:hover{opacity:.85;transform:translateY(-1px)}._addBtn_1uqg7_60:active{transform:translateY(0)}._cutter_1uqg7_79{background:#e9456026;color:#e94560;border:1px solid rgba(233,69,96,.4)}._stitcher_1uqg7_85{background:#53348333;color:#a371f7;border:1px solid rgba(163,113,247,.4)}._outputFolder_1uqg7_91{background:#2ea04326;color:#3fb950;border:1px solid rgba(46,160,67,.4)}._inputFile_1uqg7_97{background:#388bfd26;color:#388bfd;border:1px solid rgba(56,139,253,.4)}._inputFolder_1uqg7_103{background:#d2992226;color:#d29922;border:1px solid rgba(210,153,34,.4)}._saveBtn_1uqg7_109{background:#21262d;border:1px solid #30363d;border-radius:6px;color:#c9d1d9;font-size:12px;font-weight:600;padding:6px 14px;cursor:pointer;position:relative;transition:background .15s,border-color .15s,color .15s;display:flex;align-items:center;gap:6px}._saveBtn_1uqg7_109:hover:not(:disabled){background:#30363d;border-color:#8b949e}._saveBtn_1uqg7_109:disabled{opacity:.6;cursor:default}._dirty_1uqg7_135{border-color:#e94560;color:#e94560}._savedGreen_1uqg7_140{color:#3fb950;border-color:#3fb95066}._errorRed_1uqg7_145{color:#f85149;border-color:#f8514966}._serverStatus_1uqg7_150{display:flex;align-items:center;gap:5px;font-size:11px;font-weight:600;color:#484f58;transition:color .3s}._serverStatus_1uqg7_150._online_1uqg7_160{color:#3fb950}._serverStatus_1uqg7_150._offline_1uqg7_161{color:#f85149}._serverDot_1uqg7_163{width:7px;height:7px;border-radius:50%;flex-shrink:0;background:currentColor;transition:box-shadow .3s}._serverStatus_1uqg7_150._online_1uqg7_160 ._serverDot_1uqg7_163{box-shadow:0 0 5px #3fb950b3}._serverStatus_1uqg7_150._offline_1uqg7_161 ._serverDot_1uqg7_163{box-shadow:0 0 5px #f85149b3}._serverLabel_1uqg7_175{white-space:nowrap}._saveIcon_1uqg7_179{width:13px;height:13px;flex-shrink:0}._node_9qeze_1{background:#16213e;border:1.5px solid #0f3460;border-radius:10px;min-width:260px;max-width:320px;box-shadow:0 4px 20px #00000080;font-size:12px;color:#c9d1d9;transition:border-color .15s;position:relative}._deleteBtn_9qeze_14{position:absolute;top:-8px;right:-8px;width:18px;height:18px;border-radius:50%;background:#21262d;border:1.5px solid #30363d;color:#6e7681;font-size:11px;line-height:1;cursor:pointer;display:flex;align-items:center;justify-content:center;padding:0;opacity:0;transition:opacity .15s,background .15s,color .15s;z-index:10}._node_9qeze_1:hover ._deleteBtn_9qeze_14{opacity:1}._deleteBtn_9qeze_14:hover{background:#e94560;border-color:#e94560;color:#fff}._node_9qeze_1._selected_9qeze_46{border-color:#e94560;box-shadow:0 0 0 2px #e9456040,0 4px 20px #00000080}._cutter_9qeze_51{border-top:3px solid #e94560}._stitcher_9qeze_52{border-top:3px solid #533483}._outputFolder_9qeze_53{border-top:3px solid #2ea043}._inputFile_9qeze_54{border-top:3px solid #388bfd}._inputFolder_9qeze_55{border-top:3px solid #d29922}._header_9qeze_57{display:flex;align-items:center;gap:6px;padding:8px 10px 6px;border-bottom:1px solid #0f3460}._badge_9qeze_65{font-size:10px;font-weight:700;letter-spacing:.03em;color:#8b949e;text-transform:uppercase;white-space:nowrap}._labelInput_9qeze_74{flex:1;background:transparent;border:none;outline:none;color:#e0e0e0;font-size:13px;font-weight:600;min-width:0}._body_9qeze_85{padding:10px;display:flex;flex-direction:column;gap:5px}._fieldLabel_9qeze_92{font-size:10px;text-transform:uppercase;letter-spacing:.06em;color:#8b949e;margin-top:4px}._input_9qeze_54{width:100%;background:#0d1117;border:1px solid #21262d;border-radius:5px;color:#c9d1d9;font-size:12px;padding:5px 8px;outline:none;transition:border-color .15s}._input_9qeze_54:focus{border-color:#388bfd}._radioGroup_9qeze_116{display:flex;flex-direction:column;gap:3px}._radio_9qeze_116{display:flex;align-items:center;gap:5px;cursor:pointer;font-size:11px;color:#c9d1d9}._checkboxRow_9qeze_131{display:flex;gap:12px;margin-top:2px}._checkbox_9qeze_131{display:flex;align-items:center;gap:4px;cursor:pointer;font-size:11px;color:#c9d1d9}._sectionHeader_9qeze_146{display:flex;align-items:center;justify-content:space-between;margin-top:4px}._addBtn_9qeze_153{background:#21262d;border:1px solid #30363d;border-radius:4px;color:#8b949e;font-size:10px;padding:2px 7px;cursor:pointer;transition:background .1s}._addBtn_9qeze_153:hover{background:#30363d;color:#c9d1d9}._inputRow_9qeze_169{display:flex;flex-direction:column;gap:3px}._inputRowMain_9qeze_175{display:flex;gap:4px;align-items:center}._removeBtn_9qeze_181{background:none;border:none;color:#6e7681;cursor:pointer;font-size:12px;padding:2px 4px;border-radius:3px;flex-shrink:0;transition:color .1s}._removeBtn_9qeze_181:hover{color:#e94560}._hint_9qeze_197{font-size:10px;color:#6e7681;font-style:italic;margin-bottom:2px}._dragHandle_9qeze_204{cursor:grab;color:#6e7681;font-size:14px;padding:0 4px;flex-shrink:0;-webkit-user-select:none;user-select:none;line-height:1;touch-action:none}._dragHandle_9qeze_204:hover{color:#c9d1d9}._dragHandle_9qeze_204:active{cursor:grabbing}._inputRow_9qeze_169._dragging_9qeze_223{opacity:.35}._inputRow_9qeze_169._dropTarget_9qeze_227{border-top:2px solid #388bfd;margin-top:-2px}._inputRow_9qeze_169._dropAfter_9qeze_232{border-bottom:2px solid #388bfd;margin-bottom:-2px}._edgeItem_9qeze_237{flex:1;display:flex;align-items:center;gap:6px;background:#0d1117;border:1px solid #21262d;border-radius:5px;padding:5px 8px;color:#8b949e;font-size:12px;min-width:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}._stepperRow_9qeze_254{display:flex;gap:4px;align-items:stretch}._stepperRow_9qeze_254 ._input_9qeze_54{flex:1}._stepperBtns_9qeze_264{display:flex;flex-direction:column;gap:1px;flex-shrink:0}._stepBtn_9qeze_271{background:#21262d;border:1px solid #30363d;border-radius:3px;color:#8b949e;font-size:8px;padding:0 5px;cursor:pointer;line-height:1;flex:1;transition:background .1s,color .1s}._stepBtn_9qeze_271:hover{background:#30363d;color:#c9d1d9}._pencilBtn_9qeze_289{background:none;border:1px solid transparent;border-radius:3px;color:#6e7681;cursor:pointer;font-size:13px;padding:1px 4px;flex-shrink:0;transition:color .1s,border-color .1s;line-height:1.2}._pencilBtn_9qeze_289:hover{color:#c9d1d9;border-color:#30363d}._pencilBtn_9qeze_289._pencilActive_9qeze_307{color:#388bfd;border-color:#388bfd}._durationOverrideRow_9qeze_312{display:flex;align-items:center;gap:4px;padding-left:22px}._durationOverrideLabel_9qeze_319{font-size:10px;color:#6e7681;white-space:nowrap;flex-shrink:0}._resetBtn_9qeze_326{background:none;border:1px solid #30363d;border-radius:3px;color:#6e7681;cursor:pointer;font-size:12px;padding:1px 5px;flex-shrink:0;transition:color .1s;line-height:1.4}._resetBtn_9qeze_326:hover{color:#c9d1d9}._edgeBadge_9qeze_343{font-size:9px;font-weight:700;text-transform:uppercase;letter-spacing:.04em;color:#a371f7;background:#1a1030;border:1px solid #533483;border-radius:3px;padding:1px 4px;flex-shrink:0}._handle_9qeze_356{width:10px;height:10px;background:#533483;border:2px solid #a371f7;border-radius:50%}._container_1ymg3_1{display:flex;flex-direction:column;gap:2px;width:100%}._wrapper_1ymg3_8{display:flex;align-items:center;background:#0d1117;border:1px solid #21262d;border-radius:5px;transition:border-color .15s,background .15s;overflow:hidden;width:100%}._wrapper_1ymg3_8:focus-within{border-color:#388bfd}._wrapper_1ymg3_8._dragOver_1ymg3_23{border-color:#388bfd;background:#0d1a2d}._textInput_1ymg3_28{flex:1;background:transparent;border:none;color:#c9d1d9;font-size:12px;padding:5px 8px;outline:none;min-width:0}._textInput_1ymg3_28::placeholder{color:#484f58}._browseBtn_1ymg3_43{background:none;border:none;border-left:1px solid #21262d;color:#6e7681;cursor:pointer;font-size:12px;padding:4px 7px;line-height:1;flex-shrink:0;transition:color .1s,background .1s}._browseBtn_1ymg3_43:hover{color:#c9d1d9;background:#21262d}._wrapper_1ymg3_8._error_1ymg3_61{border-color:#e94560}._errorMsg_1ymg3_65{font-size:10px;color:#e94560;padding:0 2px}._hidden_1ymg3_71{display:none}.app{height:100%;display:flex;flex-direction:column}.canvas-wrapper{flex:1;margin-top:49px}.react-flow__edge-path{stroke:#8b949e;stroke-width:2}.react-flow__edge.selected .react-flow__edge-path,.react-flow__edge:hover .react-flow__edge-path{stroke:#a371f7}.react-flow__connection-path{stroke:#a371f7;stroke-width:2}.react-flow__controls{background:#0d1117;border:1px solid #21262d;border-radius:6px;overflow:hidden}.react-flow__controls-button{background:#0d1117;border-bottom:1px solid #21262d;fill:#8b949e}.react-flow__controls-button:hover{background:#21262d;fill:#c9d1d9}
1
+ .react-flow{direction:ltr;--xy-edge-stroke-default: #b1b1b7;--xy-edge-stroke-width-default: 1;--xy-edge-stroke-selected-default: #555;--xy-connectionline-stroke-default: #b1b1b7;--xy-connectionline-stroke-width-default: 1;--xy-attribution-background-color-default: rgba(255, 255, 255, .5);--xy-minimap-background-color-default: #fff;--xy-minimap-mask-background-color-default: rgba(240, 240, 240, .6);--xy-minimap-mask-stroke-color-default: transparent;--xy-minimap-mask-stroke-width-default: 1;--xy-minimap-node-background-color-default: #e2e2e2;--xy-minimap-node-stroke-color-default: transparent;--xy-minimap-node-stroke-width-default: 2;--xy-background-color-default: transparent;--xy-background-pattern-dots-color-default: #91919a;--xy-background-pattern-lines-color-default: #eee;--xy-background-pattern-cross-color-default: #e2e2e2;background-color:var(--xy-background-color, var(--xy-background-color-default));--xy-node-color-default: inherit;--xy-node-border-default: 1px solid #1a192b;--xy-node-background-color-default: #fff;--xy-node-group-background-color-default: rgba(240, 240, 240, .25);--xy-node-boxshadow-hover-default: 0 1px 4px 1px rgba(0, 0, 0, .08);--xy-node-boxshadow-selected-default: 0 0 0 .5px #1a192b;--xy-node-border-radius-default: 3px;--xy-handle-background-color-default: #1a192b;--xy-handle-border-color-default: #fff;--xy-selection-background-color-default: rgba(0, 89, 220, .08);--xy-selection-border-default: 1px dotted rgba(0, 89, 220, .8);--xy-controls-button-background-color-default: #fefefe;--xy-controls-button-background-color-hover-default: #f4f4f4;--xy-controls-button-color-default: inherit;--xy-controls-button-color-hover-default: inherit;--xy-controls-button-border-color-default: #eee;--xy-controls-box-shadow-default: 0 0 2px 1px rgba(0, 0, 0, .08);--xy-edge-label-background-color-default: #ffffff;--xy-edge-label-color-default: inherit;--xy-resize-background-color-default: #3367d9}.react-flow.dark{--xy-edge-stroke-default: #3e3e3e;--xy-edge-stroke-width-default: 1;--xy-edge-stroke-selected-default: #727272;--xy-connectionline-stroke-default: #b1b1b7;--xy-connectionline-stroke-width-default: 1;--xy-attribution-background-color-default: rgba(150, 150, 150, .25);--xy-minimap-background-color-default: #141414;--xy-minimap-mask-background-color-default: rgba(60, 60, 60, .6);--xy-minimap-mask-stroke-color-default: transparent;--xy-minimap-mask-stroke-width-default: 1;--xy-minimap-node-background-color-default: #2b2b2b;--xy-minimap-node-stroke-color-default: transparent;--xy-minimap-node-stroke-width-default: 2;--xy-background-color-default: #141414;--xy-background-pattern-dots-color-default: #777;--xy-background-pattern-lines-color-default: #777;--xy-background-pattern-cross-color-default: #777;--xy-node-color-default: #f8f8f8;--xy-node-border-default: 1px solid #3c3c3c;--xy-node-background-color-default: #1e1e1e;--xy-node-group-background-color-default: rgba(240, 240, 240, .25);--xy-node-boxshadow-hover-default: 0 1px 4px 1px rgba(255, 255, 255, .08);--xy-node-boxshadow-selected-default: 0 0 0 .5px #999;--xy-handle-background-color-default: #bebebe;--xy-handle-border-color-default: #1e1e1e;--xy-selection-background-color-default: rgba(200, 200, 220, .08);--xy-selection-border-default: 1px dotted rgba(200, 200, 220, .8);--xy-controls-button-background-color-default: #2b2b2b;--xy-controls-button-background-color-hover-default: #3e3e3e;--xy-controls-button-color-default: #f8f8f8;--xy-controls-button-color-hover-default: #fff;--xy-controls-button-border-color-default: #5b5b5b;--xy-controls-box-shadow-default: 0 0 2px 1px rgba(0, 0, 0, .08);--xy-edge-label-background-color-default: #141414;--xy-edge-label-color-default: #f8f8f8}.react-flow__background{background-color:var(--xy-background-color-props, var(--xy-background-color, var(--xy-background-color-default)));pointer-events:none;z-index:-1}.react-flow__container{position:absolute;width:100%;height:100%;top:0;left:0}.react-flow__pane{z-index:1}.react-flow__pane.draggable{cursor:grab}.react-flow__pane.dragging{cursor:grabbing}.react-flow__pane.selection{cursor:pointer}.react-flow__viewport{transform-origin:0 0;z-index:2;pointer-events:none}.react-flow__renderer{z-index:4}.react-flow__selection{z-index:6}.react-flow__nodesselection-rect:focus,.react-flow__nodesselection-rect:focus-visible{outline:none}.react-flow__edge-path{stroke:var(--xy-edge-stroke, var(--xy-edge-stroke-default));stroke-width:var(--xy-edge-stroke-width, var(--xy-edge-stroke-width-default));fill:none}.react-flow__connection-path{stroke:var(--xy-connectionline-stroke, var(--xy-connectionline-stroke-default));stroke-width:var(--xy-connectionline-stroke-width, var(--xy-connectionline-stroke-width-default));fill:none}.react-flow .react-flow__edges{position:absolute}.react-flow .react-flow__edges svg{overflow:visible;position:absolute;pointer-events:none}.react-flow__edge{pointer-events:visibleStroke}.react-flow__edge.selectable{cursor:pointer}.react-flow__edge.animated path{stroke-dasharray:5;animation:dashdraw .5s linear infinite}.react-flow__edge.animated path.react-flow__edge-interaction{stroke-dasharray:none;animation:none}.react-flow__edge.inactive{pointer-events:none}.react-flow__edge.selected,.react-flow__edge:focus,.react-flow__edge:focus-visible{outline:none}.react-flow__edge.selected .react-flow__edge-path,.react-flow__edge.selectable:focus .react-flow__edge-path,.react-flow__edge.selectable:focus-visible .react-flow__edge-path{stroke:var(--xy-edge-stroke-selected, var(--xy-edge-stroke-selected-default))}.react-flow__edge-textwrapper{pointer-events:all}.react-flow__edge .react-flow__edge-text{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none}.react-flow__arrowhead polyline{stroke:var(--xy-edge-stroke, var(--xy-edge-stroke-default))}.react-flow__arrowhead polyline.arrowclosed{fill:var(--xy-edge-stroke, var(--xy-edge-stroke-default))}.react-flow__connection{pointer-events:none}.react-flow__connection .animated{stroke-dasharray:5;animation:dashdraw .5s linear infinite}svg.react-flow__connectionline{z-index:1001;overflow:visible;position:absolute}.react-flow__nodes{pointer-events:none;transform-origin:0 0}.react-flow__node{position:absolute;-webkit-user-select:none;-moz-user-select:none;user-select:none;pointer-events:all;transform-origin:0 0;box-sizing:border-box;cursor:default}.react-flow__node.selectable{cursor:pointer}.react-flow__node.draggable{cursor:grab;pointer-events:all}.react-flow__node.draggable.dragging{cursor:grabbing}.react-flow__nodesselection{z-index:3;transform-origin:left top;pointer-events:none}.react-flow__nodesselection-rect{position:absolute;pointer-events:all;cursor:grab}.react-flow__handle{position:absolute;pointer-events:none;min-width:5px;min-height:5px;width:6px;height:6px;background-color:var(--xy-handle-background-color, var(--xy-handle-background-color-default));border:1px solid var(--xy-handle-border-color, var(--xy-handle-border-color-default));border-radius:100%}.react-flow__handle.connectingfrom{pointer-events:all}.react-flow__handle.connectionindicator{pointer-events:all;cursor:crosshair}.react-flow__handle-bottom{top:auto;left:50%;bottom:0;transform:translate(-50%,50%)}.react-flow__handle-top{top:0;left:50%;transform:translate(-50%,-50%)}.react-flow__handle-left{top:50%;left:0;transform:translate(-50%,-50%)}.react-flow__handle-right{top:50%;right:0;transform:translate(50%,-50%)}.react-flow__edgeupdater{cursor:move;pointer-events:all}.react-flow__pane.selection .react-flow__panel{pointer-events:none}.react-flow__panel{position:absolute;z-index:5;margin:15px}.react-flow__panel.top{top:0}.react-flow__panel.bottom{bottom:0}.react-flow__panel.top.center,.react-flow__panel.bottom.center{left:50%;transform:translate(-15px) translate(-50%)}.react-flow__panel.left{left:0}.react-flow__panel.right{right:0}.react-flow__panel.left.center,.react-flow__panel.right.center{top:50%;transform:translateY(-15px) translateY(-50%)}.react-flow__attribution{font-size:10px;background:var(--xy-attribution-background-color, var(--xy-attribution-background-color-default));padding:2px 3px;margin:0}.react-flow__attribution a{text-decoration:none;color:#999}@keyframes dashdraw{0%{stroke-dashoffset:10}}.react-flow__edgelabel-renderer{position:absolute;width:100%;height:100%;pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none;left:0;top:0}.react-flow__viewport-portal{position:absolute;width:100%;height:100%;left:0;top:0;-webkit-user-select:none;-moz-user-select:none;user-select:none}.react-flow__minimap{background:var( --xy-minimap-background-color-props, var(--xy-minimap-background-color, var(--xy-minimap-background-color-default)) )}.react-flow__minimap-svg{display:block}.react-flow__minimap-mask{fill:var( --xy-minimap-mask-background-color-props, var(--xy-minimap-mask-background-color, var(--xy-minimap-mask-background-color-default)) );stroke:var( --xy-minimap-mask-stroke-color-props, var(--xy-minimap-mask-stroke-color, var(--xy-minimap-mask-stroke-color-default)) );stroke-width:var( --xy-minimap-mask-stroke-width-props, var(--xy-minimap-mask-stroke-width, var(--xy-minimap-mask-stroke-width-default)) )}.react-flow__minimap-node{fill:var( --xy-minimap-node-background-color-props, var(--xy-minimap-node-background-color, var(--xy-minimap-node-background-color-default)) );stroke:var( --xy-minimap-node-stroke-color-props, var(--xy-minimap-node-stroke-color, var(--xy-minimap-node-stroke-color-default)) );stroke-width:var( --xy-minimap-node-stroke-width-props, var(--xy-minimap-node-stroke-width, var(--xy-minimap-node-stroke-width-default)) )}.react-flow__background-pattern.dots{fill:var( --xy-background-pattern-color-props, var(--xy-background-pattern-color, var(--xy-background-pattern-dots-color-default)) )}.react-flow__background-pattern.lines{stroke:var( --xy-background-pattern-color-props, var(--xy-background-pattern-color, var(--xy-background-pattern-lines-color-default)) )}.react-flow__background-pattern.cross{stroke:var( --xy-background-pattern-color-props, var(--xy-background-pattern-color, var(--xy-background-pattern-cross-color-default)) )}.react-flow__controls{display:flex;flex-direction:column;box-shadow:var(--xy-controls-box-shadow, var(--xy-controls-box-shadow-default))}.react-flow__controls.horizontal{flex-direction:row}.react-flow__controls-button{display:flex;justify-content:center;align-items:center;height:26px;width:26px;padding:4px;border:none;background:var(--xy-controls-button-background-color, var(--xy-controls-button-background-color-default));border-bottom:1px solid var( --xy-controls-button-border-color-props, var(--xy-controls-button-border-color, var(--xy-controls-button-border-color-default)) );color:var( --xy-controls-button-color-props, var(--xy-controls-button-color, var(--xy-controls-button-color-default)) );cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none}.react-flow__controls-button svg{width:100%;max-width:12px;max-height:12px;fill:currentColor}.react-flow__edge.updating .react-flow__edge-path{stroke:#777}.react-flow__edge-text{font-size:10px}.react-flow__node.selectable:focus,.react-flow__node.selectable:focus-visible{outline:none}.react-flow__node-input,.react-flow__node-default,.react-flow__node-output,.react-flow__node-group{padding:10px;border-radius:var(--xy-node-border-radius, var(--xy-node-border-radius-default));width:150px;font-size:12px;color:var(--xy-node-color, var(--xy-node-color-default));text-align:center;border:var(--xy-node-border, var(--xy-node-border-default));background-color:var(--xy-node-background-color, var(--xy-node-background-color-default))}.react-flow__node-input.selectable:hover,.react-flow__node-default.selectable:hover,.react-flow__node-output.selectable:hover,.react-flow__node-group.selectable:hover{box-shadow:var(--xy-node-boxshadow-hover, var(--xy-node-boxshadow-hover-default))}.react-flow__node-input.selectable.selected,.react-flow__node-input.selectable:focus,.react-flow__node-input.selectable:focus-visible,.react-flow__node-default.selectable.selected,.react-flow__node-default.selectable:focus,.react-flow__node-default.selectable:focus-visible,.react-flow__node-output.selectable.selected,.react-flow__node-output.selectable:focus,.react-flow__node-output.selectable:focus-visible,.react-flow__node-group.selectable.selected,.react-flow__node-group.selectable:focus,.react-flow__node-group.selectable:focus-visible{box-shadow:var(--xy-node-boxshadow-selected, var(--xy-node-boxshadow-selected-default))}.react-flow__node-group{background-color:var(--xy-node-group-background-color, var(--xy-node-group-background-color-default))}.react-flow__nodesselection-rect,.react-flow__selection{background:var(--xy-selection-background-color, var(--xy-selection-background-color-default));border:var(--xy-selection-border, var(--xy-selection-border-default))}.react-flow__nodesselection-rect:focus,.react-flow__nodesselection-rect:focus-visible,.react-flow__selection:focus,.react-flow__selection:focus-visible{outline:none}.react-flow__controls-button:hover{background:var( --xy-controls-button-background-color-hover-props, var(--xy-controls-button-background-color-hover, var(--xy-controls-button-background-color-hover-default)) );color:var( --xy-controls-button-color-hover-props, var(--xy-controls-button-color-hover, var(--xy-controls-button-color-hover-default)) )}.react-flow__controls-button:disabled{pointer-events:none}.react-flow__controls-button:disabled svg{fill-opacity:.4}.react-flow__controls-button:last-child{border-bottom:none}.react-flow__controls.horizontal .react-flow__controls-button{border-bottom:none;border-right:1px solid var( --xy-controls-button-border-color-props, var(--xy-controls-button-border-color, var(--xy-controls-button-border-color-default)) )}.react-flow__controls.horizontal .react-flow__controls-button:last-child{border-right:none}.react-flow__resize-control{position:absolute}.react-flow__resize-control.left,.react-flow__resize-control.right{cursor:ew-resize}.react-flow__resize-control.top,.react-flow__resize-control.bottom{cursor:ns-resize}.react-flow__resize-control.top.left,.react-flow__resize-control.bottom.right{cursor:nwse-resize}.react-flow__resize-control.bottom.left,.react-flow__resize-control.top.right{cursor:nesw-resize}.react-flow__resize-control.handle{width:5px;height:5px;border:1px solid #fff;border-radius:1px;background-color:var(--xy-resize-background-color, var(--xy-resize-background-color-default));translate:-50% -50%}.react-flow__resize-control.handle.left{left:0;top:50%}.react-flow__resize-control.handle.right{left:100%;top:50%}.react-flow__resize-control.handle.top{left:50%;top:0}.react-flow__resize-control.handle.bottom{left:50%;top:100%}.react-flow__resize-control.handle.top.left,.react-flow__resize-control.handle.bottom.left{left:0}.react-flow__resize-control.handle.top.right,.react-flow__resize-control.handle.bottom.right{left:100%}.react-flow__resize-control.line{border-color:var(--xy-resize-background-color, var(--xy-resize-background-color-default));border-width:0;border-style:solid}.react-flow__resize-control.line.left,.react-flow__resize-control.line.right{width:1px;transform:translate(-50%);top:0;height:100%}.react-flow__resize-control.line.left{left:0;border-left-width:1px}.react-flow__resize-control.line.right{left:100%;border-right-width:1px}.react-flow__resize-control.line.top,.react-flow__resize-control.line.bottom{height:1px;transform:translateY(-50%);left:0;width:100%}.react-flow__resize-control.line.top{top:0;border-top-width:1px}.react-flow__resize-control.line.bottom{border-bottom-width:1px;top:100%}.react-flow__edge-textbg{fill:var(--xy-edge-label-background-color, var(--xy-edge-label-background-color-default))}.react-flow__edge-text{fill:var(--xy-edge-label-color, var(--xy-edge-label-color-default))}._toolbar_1uqg7_1{position:absolute;top:0;left:0;right:0;z-index:10;display:flex;align-items:center;justify-content:space-between;padding:8px 14px;background:#0d1117eb;-webkit-backdrop-filter:blur(8px);backdrop-filter:blur(8px);border-bottom:1px solid #21262d}._left_1uqg7_16{display:flex;align-items:center;gap:10px}._right_1uqg7_22{display:flex;align-items:center;gap:8px}._pipelineName_1uqg7_28{font-size:13px;font-weight:700;color:#e0e0e0;letter-spacing:.02em;cursor:text}._pipelineName_1uqg7_28:hover{color:#fff}._pipelineNameInput_1uqg7_40{font-size:13px;font-weight:700;color:#e0e0e0;letter-spacing:.02em;background:transparent;border:none;border-bottom:1px solid #388bfd;outline:none;padding:0;min-width:80px;width:auto}._divider_1uqg7_54{width:1px;height:20px;background:#21262d}._addBtn_1uqg7_60{border:none;border-radius:6px;font-size:12px;font-weight:600;padding:6px 12px;cursor:pointer;transition:opacity .15s,transform .1s}._addBtn_1uqg7_60:hover{opacity:.85;transform:translateY(-1px)}._addBtn_1uqg7_60:active{transform:translateY(0)}._cutter_1uqg7_79{background:#e9456026;color:#e94560;border:1px solid rgba(233,69,96,.4)}._stitcher_1uqg7_85{background:#53348333;color:#a371f7;border:1px solid rgba(163,113,247,.4)}._outputFolder_1uqg7_91{background:#2ea04326;color:#3fb950;border:1px solid rgba(46,160,67,.4)}._inputFile_1uqg7_97{background:#388bfd26;color:#388bfd;border:1px solid rgba(56,139,253,.4)}._inputFolder_1uqg7_103{background:#d2992226;color:#d29922;border:1px solid rgba(210,153,34,.4)}._saveBtn_1uqg7_109{background:#21262d;border:1px solid #30363d;border-radius:6px;color:#c9d1d9;font-size:12px;font-weight:600;padding:6px 14px;cursor:pointer;position:relative;transition:background .15s,border-color .15s,color .15s;display:flex;align-items:center;gap:6px}._saveBtn_1uqg7_109:hover:not(:disabled){background:#30363d;border-color:#8b949e}._saveBtn_1uqg7_109:disabled{opacity:.6;cursor:default}._dirty_1uqg7_135{border-color:#e94560;color:#e94560}._savedGreen_1uqg7_140{color:#3fb950;border-color:#3fb95066}._errorRed_1uqg7_145{color:#f85149;border-color:#f8514966}._serverStatus_1uqg7_150{display:flex;align-items:center;gap:5px;font-size:11px;font-weight:600;color:#484f58;transition:color .3s}._serverStatus_1uqg7_150._online_1uqg7_160{color:#3fb950}._serverStatus_1uqg7_150._offline_1uqg7_161{color:#f85149}._serverDot_1uqg7_163{width:7px;height:7px;border-radius:50%;flex-shrink:0;background:currentColor;transition:box-shadow .3s}._serverStatus_1uqg7_150._online_1uqg7_160 ._serverDot_1uqg7_163{box-shadow:0 0 5px #3fb950b3}._serverStatus_1uqg7_150._offline_1uqg7_161 ._serverDot_1uqg7_163{box-shadow:0 0 5px #f85149b3}._serverLabel_1uqg7_175{white-space:nowrap}._saveIcon_1uqg7_179{width:13px;height:13px;flex-shrink:0}._node_cf3kk_1{background:#16213e;border:1.5px solid #0f3460;border-radius:10px;min-width:260px;max-width:320px;box-shadow:0 4px 20px #00000080;font-size:12px;color:#c9d1d9;transition:border-color .15s;position:relative}._deleteBtn_cf3kk_14{position:absolute;top:-8px;right:-8px;width:18px;height:18px;border-radius:50%;background:#21262d;border:1.5px solid #30363d;color:#6e7681;font-size:11px;line-height:1;cursor:pointer;display:flex;align-items:center;justify-content:center;padding:0;opacity:0;transition:opacity .15s,background .15s,color .15s;z-index:10}._node_cf3kk_1:hover ._deleteBtn_cf3kk_14{opacity:1}._deleteBtn_cf3kk_14:hover{background:#e94560;border-color:#e94560;color:#fff}._node_cf3kk_1._selected_cf3kk_46{border-color:#e94560;box-shadow:0 0 0 2px #e9456040,0 4px 20px #00000080}._cutter_cf3kk_51{border-top:3px solid #e94560}._stitcher_cf3kk_52{border-top:3px solid #533483}._outputFolder_cf3kk_53{border-top:3px solid #2ea043}._inputFile_cf3kk_54{border-top:3px solid #388bfd}._inputFolder_cf3kk_55{border-top:3px solid #d29922}._header_cf3kk_57{display:flex;align-items:center;gap:6px;padding:8px 10px 6px;border-bottom:1px solid #0f3460}._badge_cf3kk_65{font-size:10px;font-weight:700;letter-spacing:.03em;color:#8b949e;text-transform:uppercase;white-space:nowrap}._labelInput_cf3kk_74{flex:1;background:transparent;border:none;outline:none;color:#e0e0e0;font-size:13px;font-weight:600;min-width:0}._body_cf3kk_85{padding:10px;display:flex;flex-direction:column;gap:5px}._fieldLabel_cf3kk_92{font-size:10px;text-transform:uppercase;letter-spacing:.06em;color:#8b949e;margin-top:4px}._input_cf3kk_54{width:100%;background:#0d1117;border:1px solid #21262d;border-radius:5px;color:#c9d1d9;font-size:12px;padding:5px 8px;outline:none;transition:border-color .15s}._input_cf3kk_54:focus{border-color:#388bfd}._radioGroup_cf3kk_116{display:flex;flex-direction:column;gap:3px}._radio_cf3kk_116{display:flex;align-items:center;gap:5px;cursor:pointer;font-size:11px;color:#c9d1d9}._checkboxRow_cf3kk_131{display:flex;gap:12px;margin-top:2px}._checkbox_cf3kk_131{display:flex;align-items:center;gap:4px;cursor:pointer;font-size:11px;color:#c9d1d9}._sectionHeader_cf3kk_146{display:flex;align-items:center;justify-content:space-between;margin-top:4px}._addBtn_cf3kk_153{background:#21262d;border:1px solid #30363d;border-radius:4px;color:#8b949e;font-size:10px;padding:2px 7px;cursor:pointer;transition:background .1s}._addBtn_cf3kk_153:hover{background:#30363d;color:#c9d1d9}._inputRow_cf3kk_169{display:flex;flex-direction:column;gap:3px}._inputRowMain_cf3kk_175{display:flex;gap:4px;align-items:center}._removeBtn_cf3kk_181{background:none;border:none;color:#6e7681;cursor:pointer;font-size:12px;padding:2px 4px;border-radius:3px;flex-shrink:0;transition:color .1s}._removeBtn_cf3kk_181:hover{color:#e94560}._hint_cf3kk_197{font-size:10px;color:#6e7681;font-style:italic;margin-bottom:2px}._dragHandle_cf3kk_204{cursor:grab;color:#6e7681;font-size:14px;padding:0 4px;flex-shrink:0;-webkit-user-select:none;user-select:none;line-height:1;touch-action:none}._dragHandle_cf3kk_204:hover{color:#c9d1d9}._dragHandle_cf3kk_204:active{cursor:grabbing}._inputRow_cf3kk_169._dragging_cf3kk_223{opacity:.35}._inputRow_cf3kk_169._dropTarget_cf3kk_227{border-top:2px solid #388bfd;margin-top:-2px}._inputRow_cf3kk_169._dropAfter_cf3kk_232{border-bottom:2px solid #388bfd;margin-bottom:-2px}._edgeItem_cf3kk_237{flex:1;display:flex;align-items:center;gap:6px;background:#0d1117;border:1px solid #21262d;border-radius:5px;padding:5px 8px;color:#8b949e;font-size:12px;min-width:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}._stepperRow_cf3kk_254{display:flex;gap:4px;align-items:stretch}._stepperRow_cf3kk_254 ._input_cf3kk_54{flex:1}._stepperBtns_cf3kk_264{display:flex;flex-direction:column;gap:1px;flex-shrink:0}._stepBtn_cf3kk_271{background:#21262d;border:1px solid #30363d;border-radius:3px;color:#8b949e;font-size:8px;padding:0 5px;cursor:pointer;line-height:1;flex:1;transition:background .1s,color .1s}._stepBtn_cf3kk_271:hover{background:#30363d;color:#c9d1d9}._pencilBtn_cf3kk_289{background:none;border:1px solid transparent;border-radius:3px;color:#6e7681;cursor:pointer;font-size:13px;padding:1px 4px;flex-shrink:0;transition:color .1s,border-color .1s;line-height:1.2}._pencilBtn_cf3kk_289:hover{color:#c9d1d9;border-color:#30363d}._pencilBtn_cf3kk_289._pencilActive_cf3kk_307{color:#388bfd;border-color:#388bfd}._durationOverrideRow_cf3kk_312{display:flex;align-items:center;gap:4px;padding-left:22px}._durationOverrideLabel_cf3kk_319{font-size:10px;color:#6e7681;white-space:nowrap;flex-shrink:0}._resetBtn_cf3kk_326{background:none;border:1px solid #30363d;border-radius:3px;color:#6e7681;cursor:pointer;font-size:12px;padding:1px 5px;flex-shrink:0;transition:color .1s;line-height:1.4}._resetBtn_cf3kk_326:hover{color:#c9d1d9}._seqLabelPanel_cf3kk_343{padding-left:22px;display:flex;flex-direction:column;gap:4px}._seqLabelRow_cf3kk_350{display:flex;align-items:center;gap:6px}._seqLabelKey_cf3kk_356{font-size:10px;color:#6e7681;white-space:nowrap;flex-shrink:0;min-width:60px}._seqLabelInput_cf3kk_364{flex:1;min-width:0}._seqLabelToggle_cf3kk_369{display:flex;align-items:center;gap:5px;font-size:11px;color:#c9d1d9;cursor:pointer;-webkit-user-select:none;user-select:none}._seqLabelToggle_cf3kk_369 input[type=checkbox]{accent-color:#388bfd;cursor:pointer}._seqLabelWarn_cf3kk_384{font-size:10px;color:#e3b341;line-height:1.4}._edgeBadge_cf3kk_390{font-size:9px;font-weight:700;text-transform:uppercase;letter-spacing:.04em;color:#a371f7;background:#1a1030;border:1px solid #533483;border-radius:3px;padding:1px 4px;flex-shrink:0}._handle_cf3kk_403{width:10px;height:10px;background:#533483;border:2px solid #a371f7;border-radius:50%}._container_1ymg3_1{display:flex;flex-direction:column;gap:2px;width:100%}._wrapper_1ymg3_8{display:flex;align-items:center;background:#0d1117;border:1px solid #21262d;border-radius:5px;transition:border-color .15s,background .15s;overflow:hidden;width:100%}._wrapper_1ymg3_8:focus-within{border-color:#388bfd}._wrapper_1ymg3_8._dragOver_1ymg3_23{border-color:#388bfd;background:#0d1a2d}._textInput_1ymg3_28{flex:1;background:transparent;border:none;color:#c9d1d9;font-size:12px;padding:5px 8px;outline:none;min-width:0}._textInput_1ymg3_28::placeholder{color:#484f58}._browseBtn_1ymg3_43{background:none;border:none;border-left:1px solid #21262d;color:#6e7681;cursor:pointer;font-size:12px;padding:4px 7px;line-height:1;flex-shrink:0;transition:color .1s,background .1s}._browseBtn_1ymg3_43:hover{color:#c9d1d9;background:#21262d}._wrapper_1ymg3_8._error_1ymg3_61{border-color:#e94560}._errorMsg_1ymg3_65{font-size:10px;color:#e94560;padding:0 2px}._hidden_1ymg3_71{display:none}.app{height:100%;display:flex;flex-direction:column}.canvas-wrapper{flex:1;margin-top:49px}.react-flow__edge-path{stroke:#8b949e;stroke-width:2}.react-flow__edge.selected .react-flow__edge-path,.react-flow__edge:hover .react-flow__edge-path{stroke:#a371f7}.react-flow__connection-path{stroke:#a371f7;stroke-width:2}.react-flow__controls{background:#0d1117;border:1px solid #21262d;border-radius:6px;overflow:hidden}.react-flow__controls-button{background:#0d1117;border-bottom:1px solid #21262d;fill:#8b949e}.react-flow__controls-button:hover{background:#21262d;fill:#c9d1d9}
@@ -9,8 +9,8 @@
9
9
  html, body, #root { height: 100%; width: 100%; overflow: hidden; }
10
10
  body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif; background: #1a1a2e; color: #e0e0e0; }
11
11
  </style>
12
- <script type="module" crossorigin src="./assets/index-DyAG6YEF.js"></script>
13
- <link rel="stylesheet" crossorigin href="./assets/index-K7ALkYj6.css">
12
+ <script type="module" crossorigin src="./assets/index-BqNsIbxc.js"></script>
13
+ <link rel="stylesheet" crossorigin href="./assets/index-Df9WKIy8.css">
14
14
  </head>
15
15
  <body>
16
16
  <div id="root"></div>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@shaztech/video-pipeline",
3
- "version": "1.0.4",
3
+ "version": "1.1.1",
4
4
  "description": "Visual node-based video processing pipeline CLI",
5
5
  "license": "Apache-2.0",
6
6
  "repository": {
@@ -0,0 +1,131 @@
1
+ /*
2
+ * Copyright 2026 Shazron Abdullah
3
+ *
4
+ * Licensed under the Apache License, Version 2.0 (the "License");
5
+ * you may not use this file except in compliance with the License.
6
+ * You may obtain a copy of the License at
7
+ *
8
+ * http://www.apache.org/licenses/LICENSE-2.0
9
+ *
10
+ * Unless required by applicable law or agreed to in writing, software
11
+ * distributed under the License is distributed on an "AS IS" BASIS,
12
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
+ * See the License for the specific language governing permissions and
14
+ * limitations under the License.
15
+ */
16
+
17
+ import path from 'path'
18
+ import { existsSync, writeFileSync } from 'fs'
19
+ import chalk from 'chalk'
20
+ import { run } from '../runner.js'
21
+
22
+ /**
23
+ * Escapes a file path for use inside an ffmpeg filter string value.
24
+ * In ffmpeg filter syntax the following characters must be escaped with a backslash:
25
+ * \ : '
26
+ * (The path is embedded as a value in drawtext=fontfile=...:textfile=...)
27
+ */
28
+ function escapeFilterPath(p) {
29
+ // Replace backslashes first, then colons, then single-quotes
30
+ return p.replace(/\\/g, '\\\\').replace(/:/g, '\\:').replace(/'/g, "\\'")
31
+ }
32
+
33
+ /**
34
+ * Burns a sequence label (e.g. "scene 3/10") into the bottom-right corner of
35
+ * an image using ffmpeg drawtext.
36
+ *
37
+ * @param {string} srcPath - absolute path to the source image
38
+ * @param {object} opts
39
+ * @param {number} opts.index - 1-based sequence index
40
+ * @param {number} opts.total - total item count
41
+ * @param {string} [opts.prefix] - optional prefix, e.g. "scene" → "scene 3/10"
42
+ * @param {string} opts.fontFile - path to a .ttf/.otf/.ttc font file (required)
43
+ * @param {number} [opts.fontSize=48]
44
+ * @param {string} [opts.fontColor='white']
45
+ * @param {boolean} [opts.box=false] - draw a semi-transparent background box
46
+ * @param {string} [opts.boxColor='black@0.5']
47
+ * @param {number} [opts.padding=20] - px distance from right & bottom edges
48
+ * @param {string} opts.destPath - absolute path to write the annotated image
49
+ * @param {string} [opts.label] - display label for the run() spinner
50
+ * @param {boolean} [opts.dryRun=false]
51
+ */
52
+ export async function annotateImageWithSequence(srcPath, {
53
+ index,
54
+ total,
55
+ prefix,
56
+ fontFile,
57
+ fontSize = 48,
58
+ fontColor = 'white',
59
+ box = false,
60
+ boxColor = 'black@0.5',
61
+ padding = 20,
62
+ destPath,
63
+ label,
64
+ dryRun = false,
65
+ }) {
66
+ if (!fontFile) {
67
+ throw new Error(
68
+ `sequenceLabel: fontFile is required but was not provided (item: "${srcPath}")`
69
+ )
70
+ }
71
+
72
+ if (!dryRun && !existsSync(fontFile)) {
73
+ throw new Error(
74
+ `sequenceLabel: fontFile not found: ${fontFile}`
75
+ )
76
+ }
77
+
78
+ // Compose label text
79
+ const text = prefix ? `${prefix} ${index}/${total}` : `${index}/${total}`
80
+
81
+ console.log(
82
+ chalk.dim(` [seq-label] Annotating ${path.basename(srcPath)} `) +
83
+ chalk.cyan(`"${text}"`) +
84
+ chalk.dim(` → ${path.basename(destPath)}`)
85
+ )
86
+
87
+ // Write text to a sidecar file so we don't have to worry about ffmpeg
88
+ // filter-string escaping of the text itself (handles apostrophes, colons, etc.)
89
+ const textFile = `${destPath}.txt`
90
+
91
+ if (!dryRun) {
92
+ console.log(chalk.dim(` [seq-label] text file: ${textFile}`))
93
+ writeFileSync(textFile, text, 'utf8')
94
+ console.log(chalk.dim(` [seq-label] font file: ${fontFile}`))
95
+ console.log(chalk.dim(` [seq-label] dest path: ${destPath}`))
96
+ }
97
+
98
+ // Build drawtext filter. x/y place the text padding px from right/bottom edges.
99
+ // w, h, tw, th are built-in ffmpeg drawtext variables.
100
+ const escapedFontFile = escapeFilterPath(fontFile)
101
+ const escapedTextFile = escapeFilterPath(textFile)
102
+ const x = `w-tw-${padding}`
103
+ const y = `h-th-${padding}`
104
+
105
+ let filterStr =
106
+ `drawtext=fontfile=${escapedFontFile}` +
107
+ `:textfile=${escapedTextFile}` +
108
+ `:fontsize=${fontSize}` +
109
+ `:fontcolor=${fontColor}` +
110
+ `:x=${x}` +
111
+ `:y=${y}`
112
+
113
+ if (box) {
114
+ filterStr += `:box=1:boxcolor=${boxColor}:boxborderw=8`
115
+ }
116
+
117
+ const argv = [
118
+ '-nostdin', // prevent ffmpeg from reading stdin (avoids hangs in pipelines)
119
+ '-loglevel', 'warning', // suppress verbose progress output
120
+ '-y',
121
+ '-i', srcPath,
122
+ '-vf', filterStr,
123
+ '-frames:v', '1',
124
+ destPath,
125
+ ]
126
+
127
+ await run('ffmpeg', argv, {
128
+ label: label ?? `annotate ${index}/${total}`,
129
+ dryRun,
130
+ })
131
+ }
@@ -18,6 +18,13 @@ import os from 'os'
18
18
  import path from 'path'
19
19
  import { mkdirSync, existsSync, copyFileSync } from 'fs'
20
20
  import { run } from '../runner.js'
21
+ import { annotateImageWithSequence } from './imageAnnotate.js'
22
+
23
+ const IMAGE_EXTS = /\.(png|jpe?g|gif|webp|bmp|tiff?|avif|svg)$/i
24
+
25
+ function isImage(filePath) {
26
+ return IMAGE_EXTS.test(filePath)
27
+ }
21
28
 
22
29
  function expandPath(p) {
23
30
  return p.startsWith('~/') ? path.join(os.homedir(), p.slice(2)) : p
@@ -25,7 +32,8 @@ function expandPath(p) {
25
32
 
26
33
  /**
27
34
  * Builds the list of per-run inputs from inputOrder + edge outputs.
28
- * Returns an array of { inputs: string[], name: string } one entry per output file.
35
+ * Returns { runs, runCount } where each run is:
36
+ * { inputs: Array<{ value: string, imageDuration?: number, sequenceLabel?: object }>, name: string }
29
37
  *
30
38
  * When an edge item expands to N files, N runs are produced (one per edge file),
31
39
  * with fixed items applied to every run. The output filename equals the basename
@@ -51,8 +59,8 @@ function buildRuns(config, incomingEdges, context, nodeId) {
51
59
  // No edge connections — single run with all fixed inputs
52
60
  const inputs = config.inputOrder
53
61
  .filter((i) => i.type === 'fixed' && i.value)
54
- .map((i) => i.imageDuration != null ? `${i.value}:${i.imageDuration}` : i.value)
55
- return [{ inputs, name: 'output.mp4' }]
62
+ .map((i) => ({ value: i.value, imageDuration: i.imageDuration, sequenceLabel: i.sequenceLabel }))
63
+ return { runs: [{ inputs, name: 'output.mp4' }], runCount: 1 }
56
64
  }
57
65
 
58
66
  // Number of runs = max file count across all edge items
@@ -75,12 +83,14 @@ function buildRuns(config, incomingEdges, context, nodeId) {
75
83
  const inputs = []
76
84
  for (const item of config.inputOrder) {
77
85
  if (item.type === 'fixed') {
78
- if (item.value) inputs.push(item.imageDuration != null ? `${item.value}:${item.imageDuration}` : item.value)
86
+ if (item.value) {
87
+ inputs.push({ value: item.value, imageDuration: item.imageDuration, sequenceLabel: item.sequenceLabel })
88
+ }
79
89
  } else if (item.type === 'edge') {
80
90
  const files = edgeOutputs.get(item.nodeId) ?? []
81
91
  // Use file i; clamp to last available if this edge has fewer files
82
92
  const file = files[Math.min(i, files.length - 1)]
83
- if (file) inputs.push(file)
93
+ if (file) inputs.push({ value: file })
84
94
  }
85
95
  }
86
96
  const pivotFile = pivotFiles[i]
@@ -93,7 +103,7 @@ function buildRuns(config, incomingEdges, context, nodeId) {
93
103
  : `output_${String(i + 1).padStart(3, '0')}.mp4`
94
104
  runs.push({ inputs, name })
95
105
  }
96
- return runs
106
+ return { runs, runCount }
97
107
  } else {
98
108
  // Legacy: fixed inputs first, then all edge outputs — single run
99
109
  const fixedInputs = Array.isArray(config.inputs) ? config.inputs.filter(Boolean) : []
@@ -107,10 +117,20 @@ function buildRuns(config, incomingEdges, context, nodeId) {
107
117
  }
108
118
  variableInputs.push(...(sourceCtx.outputs ?? []))
109
119
  }
110
- return [{ inputs: [...fixedInputs, ...variableInputs], name: 'output.mp4' }]
120
+ const inputs = [...fixedInputs, ...variableInputs].map((v) => ({ value: v }))
121
+ return { runs: [{ inputs, name: 'output.mp4' }], runCount: 1 }
111
122
  }
112
123
  }
113
124
 
125
+ /**
126
+ * Flattens an input object to the CLI string expected by video-stitcher:
127
+ * - image with duration → "path:duration"
128
+ * - otherwise → "path"
129
+ */
130
+ function flattenInput({ value, imageDuration }) {
131
+ return imageDuration != null ? `${value}:${imageDuration}` : value
132
+ }
133
+
114
134
  /**
115
135
  * Builds argv and executes video-stitcher for a node.
116
136
  * Produces one output file per edge segment (N inputs → N outputs),
@@ -135,11 +155,13 @@ export async function handleVideoStitcher(node, context, tempRoot, incomingEdges
135
155
 
136
156
  if (!opts.dryRun) mkdirSync(outputDir, { recursive: true })
137
157
 
138
- const runs = buildRuns(config, incomingEdges, context, node.id)
158
+ const { runs, runCount } = buildRuns(config, incomingEdges, context, node.id)
139
159
 
140
160
  const outputFiles = []
141
161
 
142
- for (const { inputs, name } of runs) {
162
+ for (let runIdx = 0; runIdx < runs.length; runIdx++) {
163
+ const { inputs, name } = runs[runIdx]
164
+
143
165
  if (inputs.length < 2) {
144
166
  throw new Error(
145
167
  `Node "${node.id}" (video-stitcher): at least 2 inputs required for "${name}", got ${inputs.length}`
@@ -156,7 +178,40 @@ export async function handleVideoStitcher(node, context, tempRoot, incomingEdges
156
178
  )
157
179
  }
158
180
 
159
- const argv = [...inputs, '-o', outputFile]
181
+ // Pre-process any fixed image inputs that have sequenceLabel.enabled
182
+ const resolvedInputs = await Promise.all(
183
+ inputs.map(async (input) => {
184
+ const sl = input.sequenceLabel
185
+ if (!sl?.enabled || !isImage(input.value)) return input
186
+
187
+ const annotDir = path.join(tempRoot, node.id, 'annotated')
188
+ if (!opts.dryRun) mkdirSync(annotDir, { recursive: true })
189
+
190
+ const ext = path.extname(input.value)
191
+ const base = path.basename(input.value, ext)
192
+ const destPath = path.join(annotDir, `${runIdx + 1}_${base}${ext}`)
193
+
194
+ await annotateImageWithSequence(input.value, {
195
+ index: runIdx + 1,
196
+ total: runCount,
197
+ prefix: sl.prefix,
198
+ fontFile: sl.fontFile,
199
+ fontSize: sl.fontSize,
200
+ fontColor: sl.fontColor,
201
+ box: sl.box,
202
+ boxColor: sl.boxColor,
203
+ padding: sl.padding,
204
+ destPath,
205
+ label: `${node.label ?? node.id} [annotate ${runIdx + 1}/${runCount}]`,
206
+ dryRun: opts.dryRun,
207
+ })
208
+
209
+ return { ...input, value: destPath }
210
+ })
211
+ )
212
+
213
+ const cliInputs = resolvedInputs.map(flattenInput)
214
+ const argv = [...cliInputs, '-o', outputFile]
160
215
 
161
216
  if (config.imageDuration != null && config.imageDuration !== 1) {
162
217
  argv.push('-d', String(config.imageDuration))