@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.
- package/dist/editor/assets/index-BqNsIbxc.js +71 -0
- package/dist/editor/assets/{index-K7ALkYj6.css → index-Df9WKIy8.css} +1 -1
- package/dist/editor/index.html +2 -2
- package/package.json +1 -1
- package/src/executor/nodeHandlers/imageAnnotate.js +131 -0
- package/src/executor/nodeHandlers/video-stitcher.js +65 -10
- package/dist/editor/assets/index-DyAG6YEF.js +0 -71
|
@@ -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}
|
package/dist/editor/index.html
CHANGED
|
@@ -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-
|
|
13
|
-
<link rel="stylesheet" crossorigin href="./assets/index-
|
|
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
|
@@ -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
|
|
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) =>
|
|
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)
|
|
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
|
-
|
|
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 (
|
|
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
|
-
|
|
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))
|