super-hero/extensions/shader-graph/dist/block-forge/forge.js

616 lines
81 KiB
JavaScript

'use strict';
Object.defineProperty(exports, "__esModule", { value: true });
exports.HTMLGraphForgeElement = void 0;
const tslib_1 = require("tslib");
const graph_1 = require("./graph");
const utils_1 = require("./utils");
const event_1 = require("./event");
const structures_1 = require("@itharbors/structures");
const undo_1 = require("./undo");
const js_yaml_1 = tslib_1.__importDefault(require("js-yaml"));
const enum_1 = require("./enum");
const STYLE = /*css*/ `
:host { display: flex; flex-direction: column; }
:host > header { padding: 4px 10px; display: flex; }
:host > header > div { flex: 1; }
:host > header > div > span { cursor: pointer; }
:host > header > slot { display: block; }
:host > header > i { margin: 0 4px; }
:host > section { flex: 1; display: flex; }
:host > section > v-graph { flex: 1; }
`;
const HTML = /*html*/ `
<style>${STYLE}</style>
<header>
<div></div>
<slot></slot>
</header>
<section>
<v-graph type=""><v-graph>
</section>
`;
class HTMLGraphForgeElement extends HTMLElement {
constructor() {
super();
this.actionQueue = new structures_1.ActionQueue({
forge: this,
});
this.paths = [];
this.attachShadow({
mode: 'open',
});
this.shadowRoot.innerHTML = HTML;
this.$graph = this.shadowRoot.querySelector('v-graph');
this._initHeader();
this._initSection();
}
_initHeader() {
this._updateHeader();
this.shadowRoot.querySelector('header > div').addEventListener('click', (event) => {
const $span = event.target;
if (!$span.hasAttribute('path-index')) {
return;
}
let index = parseInt($span.getAttribute('path-index') || '0');
if (index < 0) {
index = 0;
}
this.paths.splice(index + 1);
this._updateGraph();
const graph = this.paths[this.paths.length - 1];
(0, utils_1.dispatch)(this, 'enter-graph', {
detail: {
id: graph.name,
},
});
});
}
_updateHeader() {
const paths = this.paths.map((info, index) => `<span path-index="${index}">${info.name || info.type}</span>`).join('<i>/</i>');
this.shadowRoot.querySelector('header > div').innerHTML = paths;
}
_initSection() {
const $graph = this.shadowRoot.querySelector('v-graph');
$graph.shadowRoot.addEventListener('block-click', (event) => {
const customEvent = event;
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $node = customEvent.target;
if (info.graph.event && info.graph.event.onBlockClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $node.getAttribute('node-uuid') || '';
const block = $graph.getProperty('nodes')[uuid];
const blockEvent = new event_1.BlockMouseEvent(nodes, lines, $node, block);
info.graph.event.onBlockClick(blockEvent);
}
});
$graph.shadowRoot.addEventListener('block-dblclick', (event) => {
const customEvent = event;
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $node = customEvent.target;
if ($node.tagName === 'V-GRAPH-NODE') {
const details = $node.getProperty('details');
if (details.subGraph) {
this.enterSubGraph(details.subGraph);
return;
}
}
if (info.graph.event && info.graph.event.onBlockDblClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $node.getAttribute('node-uuid') || '';
const block = $graph.getProperty('nodes')[uuid];
const blockEvent = new event_1.BlockMouseEvent(nodes, lines, $node, block);
blockEvent.initPagePosition(customEvent.detail.pageX, customEvent.detail.pageY);
const graphPosition = $graph.convertCoordinate(customEvent.detail.offsetX, customEvent.detail.offsetY);
blockEvent.initGraphPosition(graphPosition.x, graphPosition.y);
info.graph.event.onBlockDblClick(blockEvent);
}
});
$graph.shadowRoot.addEventListener('block-right-click', (event) => {
const customEvent = event;
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $node = customEvent.target;
if (info.graph.event && info.graph.event.onBlockRightClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $node.getAttribute('node-uuid') || '';
const block = $graph.getProperty('nodes')[uuid];
const blockEvent = new event_1.BlockMouseEvent(nodes, lines, $node, block);
info.graph.event.onBlockRightClick(blockEvent);
}
});
$graph.addEventListener('node-selected', (event) => {
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $node = event.target;
if (info.graph.event && info.graph.event.onBlockSelected) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $node.getAttribute('node-uuid') || '';
const block = $graph.getProperty('nodes')[uuid];
const event = new event_1.BlockEvent(nodes, lines, $node, block);
info.graph.event.onBlockSelected(event);
}
});
$graph.addEventListener('node-unselected', (event) => {
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $node = event.target;
if (info.graph.event && info.graph.event.onBlockUnselected) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $node.getAttribute('node-uuid') || '';
const block = $graph.getProperty('nodes')[uuid];
const event = new event_1.BlockEvent(nodes, lines, $node, block);
info.graph.event.onBlockUnselected(event);
}
});
$graph.addEventListener('line-selected', (event) => {
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $g = event.target;
if (info.graph.event && info.graph.event.onLineSelected) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $g.getAttribute('line-uuid') || '';
const line = lines[uuid];
const event = new event_1.LineEvent(nodes, lines, $g, line);
info.graph.event.onLineSelected(event);
}
});
$graph.addEventListener('line-unselected', (event) => {
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
const $g = event.target;
if (info.graph.event && info.graph.event.onLineUnselected) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $g.getAttribute('line-uuid') || '';
const line = lines[uuid];
const event = new event_1.LineEvent(nodes, lines, $g, line);
info.graph.event.onLineUnselected(event);
}
});
$graph.addEventListener('node-added', (event) => {
const cEvent = event;
(0, utils_1.dispatch)(this, 'node-added', {
detail: cEvent.detail,
});
(0, utils_1.dispatch)(this, 'dirty');
});
$graph.addEventListener('node-removed', (event) => {
const cEvent = event;
(0, utils_1.dispatch)(this, 'node-removed', {
detail: cEvent.detail,
});
(0, utils_1.dispatch)(this, 'dirty');
});
$graph.addEventListener('node-changed', (event) => {
const cEvent = event;
(0, utils_1.dispatch)(this, 'node-changed', {
detail: cEvent.detail,
});
(0, utils_1.dispatch)(this, 'dirty');
});
$graph.addEventListener('node-position-changed', (event) => {
const cEvent = event;
const queue = cEvent.detail.moveList.map((item) => {
return new undo_1.BlockPositionAction({
blockName: item.id,
target: item.target,
source: item.source,
});
});
if (queue.length === 1) {
this.actionQueue.exec(queue[0]);
}
else if (queue.length > 1) {
this.actionQueue.exec(new structures_1.ActionList({
queue,
}));
}
(0, utils_1.dispatch)(this, 'dirty', {
detail: {
dirtyType: 'position-changed',
},
});
});
// //// ////
$graph.shadowRoot.addEventListener('dirty', (event) => {
const cEvent = event;
if (cEvent.detail && cEvent.detail.action) {
this.actionQueue.exec(cEvent.detail.action);
}
(0, utils_1.dispatch)(this, 'dirty', {
detail: cEvent.detail,
});
});
$graph.addEventListener('mouseup', (event) => {
const info = graph_1.graphMap.get(this.rootGraph.type);
if (!info) {
return;
}
if (event.button === 2 && info.graph.event?.onGraphRightClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const graphPosition = $graph.convertCoordinate(event.offsetX, event.offsetY);
const customEvent = new event_1.GraphMouseEvent(nodes, lines, $graph, this);
customEvent.initPagePosition(event.pageX, event.pageY);
customEvent.initGraphPosition(graphPosition.x, graphPosition.y);
info.graph.event.onGraphRightClick(customEvent);
}
});
$graph.addEventListener('line-added', (event) => {
const customEment = event;
const $node = $graph.queryNodeElement(customEment.detail.line.output.node);
if ($node) {
// @ts-ignore
$node.onUpdate && $node.onUpdate();
}
(0, utils_1.dispatch)(this, 'line-added', {
detail: customEment.detail,
});
(0, utils_1.dispatch)(this, 'dirty');
});
$graph.addEventListener('line-removed', (event) => {
const customEment = event;
const $node = $graph.queryNodeElement(customEment.detail.line.output.node);
if ($node) {
// @ts-ignore
$node.onUpdate && $node.onUpdate();
}
(0, utils_1.dispatch)(this, 'line-removed', {
detail: customEment.detail,
});
(0, utils_1.dispatch)(this, 'dirty');
});
$graph.addEventListener('line-changed', (event) => {
const customElement = event;
(0, utils_1.dispatch)(this, 'line-changed', {
detail: customElement.detail,
});
(0, utils_1.dispatch)(this, 'dirty');
});
$graph.addEventListener('node-connected', (event) => {
const customElement = event;
this.startRecording();
this.addLine(customElement.detail.line);
setTimeout(() => {
this.stopRecording();
}, 200);
});
const $svg = $graph.shadowRoot.querySelector('#lines');
function searchG(htmlArray) {
const length = Math.min(htmlArray.length, 4);
for (let i = 0; i < length; i++) {
const $elem = htmlArray[i];
// 如果找到顶部的 document 元素的话,是没有 tagName 的
if ($elem.tagName && $elem.tagName.toLocaleLowerCase() === 'g') {
return $elem;
}
}
}
$svg.addEventListener('dblclick', (event) => {
// @ts-ignore
const $g = searchG(event.path);
if (!$g || !$g.hasAttribute('line-uuid')) {
return;
}
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
if (info.graph.event && info.graph.event.onLineDblClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $g.getAttribute('line-uuid') || '';
const line = lines[uuid];
const event = new event_1.LineMouseEvent(nodes, lines, $g, line);
info.graph.event.onLineDblClick(event);
}
});
$svg.addEventListener('click', (event) => {
// @ts-ignore
const $g = searchG(event.path);
if (!$g || !$g.hasAttribute('line-uuid')) {
return;
}
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
if (info.graph.event && info.graph.event.onLineClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $g.getAttribute('line-uuid') || '';
const line = lines[uuid];
const event = new event_1.LineMouseEvent(nodes, lines, $g, line);
info.graph.event.onLineClick(event);
}
});
$svg.addEventListener('mouseup', (event) => {
// @ts-ignore
const $g = searchG(event.path);
if (!$g || !$g.hasAttribute('line-uuid')) {
return;
}
if (event.button !== 2) {
return;
}
const type = this.paths[this.paths.length - 1].type;
const info = graph_1.graphMap.get(type);
if (!info) {
return;
}
if (info.graph.event && info.graph.event.onLineRightClick) {
const nodes = $graph.getProperty('nodes');
const lines = $graph.getProperty('lines');
const uuid = $g.getAttribute('line-uuid') || '';
const line = lines[uuid];
const event = new event_1.LineMouseEvent(nodes, lines, $g, line);
info.graph.event.onLineRightClick(event);
}
});
}
_updateGraph() {
(0, enum_1.clearDynamicEnum)();
const graph = this.paths[this.paths.length - 1];
const $graph = this.shadowRoot.querySelector('v-graph');
$graph.clear();
requestAnimationFrame(() => {
$graph.setAttribute('type', graph.type);
$graph.setProperty('lines', graph.lines);
$graph.setProperty('nodes', graph.nodes);
this._updateHeader();
});
}
undo() {
this.actionQueue.undo();
(0, utils_1.dispatch)(this, 'undo');
}
redo() {
this.actionQueue.redo();
(0, utils_1.dispatch)(this, 'redo');
}
startRecording() {
this.actionQueue.startRecording();
}
stopRecording() {
this.actionQueue.stopRecording();
}
getPinElement(blockName, type, index) {
const $block = this.$graph.shadowRoot.querySelector(`v-graph-node[node-uuid=${blockName}]`);
if (!$block) {
return;
}
const $pinList = $block.shadowRoot.querySelectorAll(`.pin.in`);
const $pin = $pinList[index];
return $pin;
}
getBlockElement(blockName) {
return this.$graph.shadowRoot.querySelector(`v-graph-node[node-uuid=${blockName}]`);
}
/// ---- 操作整个图
/**
* 将屏幕坐标转换成 Graph 内的坐标
* @param point
* @returns
*/
convertCoordinate(point) {
point = this.$graph.convertCoordinate(point.x, point.y);
return point;
}
/**
* 设置编辑的根图
* @param graph
*/
setRootGraph(graph) {
this.rootGraph = graph;
this.paths = [graph];
this._updateGraph();
}
/**
* 获取正在编辑的根图
* @returns
*/
getRootGraph() {
return this.paths[0];
}
/**
* 传入一个字符串,反序列化成图数据
* @param content
* @returns
*/
deserialize(content) {
const graphData = js_yaml_1.default.load(content);
return graphData;
}
/**
* 传入一个图数据,序列化成 yaml 字符串
* @param data
* @returns
*/
serialize(data) {
const str = js_yaml_1.default.dump(data || this.paths[0]);
// return JSON.stringify(this.paths[0]);
// outputFileSync('/Users/wangsijie/Project/Creator/cocos-editor/extension-repos/shader-graph/test.yaml', str);
return str;
}
/**
* 获取整个图现在的一些基础数据
* @returns
*/
getGraphInfo() {
const offset = this.$graph.getProperty('offset');
const scale = this.$graph.getProperty('scale');
return {
offset, scale,
};
}
/**
* 设置整个图的一些基础数据
* @param info
*/
setGraphInfo(info) {
this.$graph.setProperty('offset', info.offset);
this.$graph.setProperty('scale', info.scale);
}
/**
* 恢复缩放比例
*/
zoomToFit() {
this.$graph.data.setProperty('scale', 1);
}
/// ---- 操作当前图
/**
* 获取选中的 Block 列表
* @returns
*/
getSelectedBlockList() {
return this.$graph.getSelectedNodeList();
}
/**
* 获取选中的 Line 列表
* @returns
*/
getSelectedLineList() {
return this.$graph.getSelectedLineList();
}
/**
* 设置当前正在编辑的图数据
* @param graph
* @returns
*/
setCurrentGraph(graph) {
if (this.paths.length <= 1) {
this.setRootGraph(graph);
return;
}
this.paths[this.paths.length - 1] = graph;
this._updateGraph();
}
/**
* 获取正在编辑的图数据
* @returns
*/
getCurrentGraph() {
return this.paths[this.paths.length - 1];
}
/**
* 在当前正在操作的图数据里增加一个 Block
* @param block
* @param id
*/
addBlock(block, id) {
this.actionQueue.exec(new undo_1.AddBlockAction({ block, id }));
}
/**
* 在当前正在操作的图数据里删除一个节点
* @param id
*/
removeBlock(id) {
const queue = [];
// remove line
const lines = this.$graph.getProperty('lines');
for (const key in lines) {
const line = lines[key];
if (line.input.node === id || line.output.node === id) {
queue.push(new undo_1.RemoveLineAction({ id: key }));
}
}
queue.push(new undo_1.RemoveBlockAction({ id }));
this.actionQueue.exec(new structures_1.ActionList({
queue,
}));
}
/**
* 在当前正在操作的图数据里增加一个连线
* @param line
* @param id
*/
addLine(line, id) {
this.actionQueue.exec(new undo_1.AddLineAction({ line, id }));
}
/**
* 在当前正在操作的图数据里删除一个连线
* @param id
*/
removeLine(id) {
this.actionQueue.exec(new undo_1.RemoveLineAction({ id }));
}
/**
* 进入当前图的子图
* @param id
*/
enterSubGraph(id) {
const graph = this.paths[this.paths.length - 1];
const subGraph = graph.graphs[id];
if (subGraph) {
this.paths.push(subGraph);
this._updateGraph();
}
(0, utils_1.dispatch)(this, 'enter-graph', {
detail: {
id: id,
},
});
}
/**
* 在当前编辑的图里增加一个子图
* @param type
* @param id
* @returns
*/
addSubGraph(type, id) {
const info = this.paths[this.paths.length - 1];
// const uuid = generateUUID();
info.graphs[id] = {
type,
name: type,
nodes: {},
lines: {},
graphs: {},
};
return info.graphs[id];
}
/**
* 在当前编辑的图里,删除一个子图
* @param id
*/
removeSubGraph(id) {
const info = this.paths[this.paths.length - 1];
delete info.graphs[id];
}
}
exports.HTMLGraphForgeElement = HTMLGraphForgeElement;
if (!window.customElements.get('ui-graph-forge')) {
window.customElements.define('ui-graph-forge', HTMLGraphForgeElement);
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"forge.js","sourceRoot":"","sources":["../../src/block-forge/forge.ts"],"names":[],"mappings":"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","sourcesContent":["'use strict';\n\nimport type {\n    GraphElement,\n    NodeChangedDetail,\n    NodePositionChangedDetail,\n    GraphNodeElement,\n} from '@itharbors/ui-graph';\n\nimport type { DirtyDetail } from './pin';\nimport { graphMap } from './graph';\nimport { generateUUID, dispatch } from './utils';\nimport { GraphMouseEvent, BlockMouseEvent, BlockEvent, LineEvent, LineMouseEvent } from './event';\n\nimport {\n    Action,\n    ActionList,\n    ActionQueue,\n} from '@itharbors/structures';\n\nimport {\n    AddBlockAction,\n    RemoveBlockAction,\n    AddLineAction,\n    RemoveLineAction,\n    BlockPositionAction,\n} from './undo';\n\nimport type { GraphData, BlockData, LineData, IGraphDefineEvent } from './interface';\n\nimport yaml from 'js-yaml';\nimport { clearDynamicEnum } from './enum';\n\nconst STYLE = /*css*/`\n:host { display: flex; flex-direction: column; }\n:host > header { padding: 4px 10px; display: flex; }\n:host > header > div { flex: 1; }\n:host > header > div > span { cursor: pointer; }\n:host > header > slot { display: block; }\n:host > header > i { margin: 0 4px; }\n:host > section { flex: 1; display: flex; }\n:host > section > v-graph { flex: 1; }\n`;\n\nconst HTML = /*html*/`\n<style>${STYLE}</style>\n<header>\n    <div></div>\n    <slot></slot>\n</header>\n<section>\n    <v-graph type=\"\"><v-graph>\n</section>\n`;\n\nexport class HTMLGraphForgeElement extends HTMLElement {\n\n    private actionQueue = new ActionQueue({\n        forge: this,\n    });\n\n    $graph: GraphElement;\n\n    constructor() {\n        super();\n        this.attachShadow({\n            mode: 'open',\n        });\n\n        this.shadowRoot!.innerHTML = HTML;\n\n        this.$graph = this.shadowRoot!.querySelector('v-graph')! as GraphElement;\n\n        this._initHeader();\n        this._initSection();\n    }\n\n    rootGraph?: GraphData;\n    paths: GraphData[] = [];\n\n    private _initHeader() {\n        this._updateHeader();\n        this.shadowRoot!.querySelector('header > div')!.addEventListener('click', (event) => {\n            const $span = event.target as HTMLElement;\n            if (!$span.hasAttribute('path-index')) {\n                return;\n            }\n            let index = parseInt($span.getAttribute('path-index') || '0');\n            if (index < 0) {\n                index = 0;\n            }\n            this.paths.splice(index + 1);\n            this._updateGraph();\n            const graph = this.paths[this.paths.length - 1];\n            dispatch(this, 'enter-graph', {\n                detail: {\n                    id: graph.name,\n                },\n            });\n        });\n    }\n\n    private _updateHeader() {\n        const paths = this.paths.map((info, index) => `<span path-index=\"${index}\">${info.name || info.type}</span>`).join('<i>/</i>');\n        this.shadowRoot!.querySelector('header > div')!.innerHTML = paths;\n    }\n\n    private _initSection() {\n        const $graph = this.shadowRoot!.querySelector('v-graph') as GraphElement;\n        $graph.shadowRoot!.addEventListener('block-click', (event) => {\n            const customEvent = event as CustomEvent<{\n            }>;\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $node = customEvent.target as GraphNodeElement;\n            if (info.graph.event && info.graph.event.onBlockClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $node.getAttribute('node-uuid') || '';\n                const block = $graph.getProperty('nodes')[uuid] as BlockData;\n                const blockEvent = new BlockMouseEvent(nodes, lines, $node, block);\n                info.graph.event.onBlockClick(blockEvent);\n            }\n        });\n        $graph.shadowRoot!.addEventListener('block-dblclick', (event) => {\n            const customEvent = event as CustomEvent<{\n                pageX: number;\n                pageY: number;\n                offsetX: number;\n                offsetY: number;\n            }>;\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $node = customEvent.target as GraphNodeElement;\n            if ($node.tagName === 'V-GRAPH-NODE') {\n                const details = $node.getProperty('details');\n                if (details.subGraph) {\n                    this.enterSubGraph(details.subGraph);\n                    return;\n                }\n            }\n            if (info.graph.event && info.graph.event.onBlockDblClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $node.getAttribute('node-uuid') || '';\n                const block = $graph.getProperty('nodes')[uuid] as BlockData;\n                const blockEvent = new BlockMouseEvent(nodes, lines, $node, block);\n                blockEvent.initPagePosition(customEvent.detail.pageX, customEvent.detail.pageY);\n                const graphPosition = $graph.convertCoordinate(customEvent.detail.offsetX, customEvent.detail.offsetY);\n                blockEvent.initGraphPosition(graphPosition.x, graphPosition.y);\n                info.graph.event.onBlockDblClick(blockEvent);\n            }\n        });\n        $graph.shadowRoot!.addEventListener('block-right-click', (event) => {\n            const customEvent = event as CustomEvent<{\n            }>;\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $node = customEvent.target as GraphNodeElement;\n            if (info.graph.event && info.graph.event.onBlockRightClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $node.getAttribute('node-uuid') || '';\n                const block = $graph.getProperty('nodes')[uuid] as BlockData;\n                const blockEvent = new BlockMouseEvent(nodes, lines, $node, block);\n                info.graph.event.onBlockRightClick(blockEvent);\n            }\n        });\n\n        $graph.addEventListener('node-selected', (event) => {\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $node = event.target as GraphNodeElement;\n            if (info.graph.event && info.graph.event.onBlockSelected) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $node.getAttribute('node-uuid') || '';\n                const block = $graph.getProperty('nodes')[uuid] as BlockData;\n                const event = new BlockEvent(nodes, lines, $node, block);\n                info.graph.event.onBlockSelected(event);\n            }\n        });\n        $graph.addEventListener('node-unselected', (event) => {\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $node = event.target as GraphNodeElement;\n            if (info.graph.event && info.graph.event.onBlockUnselected) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $node.getAttribute('node-uuid') || '';\n                const block = $graph.getProperty('nodes')[uuid] as BlockData;\n                const event = new BlockEvent(nodes, lines, $node, block);\n                info.graph.event.onBlockUnselected(event);\n            }\n        });\n        $graph.addEventListener('line-selected', (event) => {\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $g = event.target as SVGGElement;\n            if (info.graph.event && info.graph.event.onLineSelected) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $g.getAttribute('line-uuid') || '';\n                const line = lines[uuid] as LineData;\n                const event = new LineEvent(nodes, lines, $g, line);\n                info.graph.event.onLineSelected(event);\n            }\n        });\n        $graph.addEventListener('line-unselected', (event) => {\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            const $g = event.target as SVGGElement;\n            if (info.graph.event && info.graph.event.onLineUnselected) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $g.getAttribute('line-uuid') || '';\n                const line = lines[uuid] as LineData;\n                const event = new LineEvent(nodes, lines, $g, line);\n                info.graph.event.onLineUnselected(event);\n            }\n        });\n        $graph.addEventListener('node-added', (event) => {\n            const cEvent = event as CustomEvent<NodeChangedDetail>;\n            dispatch(this, 'node-added', {\n                detail: cEvent.detail,\n            });\n            dispatch(this, 'dirty');\n        });\n        $graph.addEventListener('node-removed', (event) => {\n            const cEvent = event as CustomEvent<NodeChangedDetail>;\n            dispatch(this, 'node-removed', {\n                detail: cEvent.detail,\n            });\n            dispatch(this, 'dirty');\n        });\n        $graph.addEventListener('node-changed', (event) => {\n            const cEvent = event as CustomEvent<NodeChangedDetail>;\n            dispatch(this, 'node-changed', {\n                detail: cEvent.detail,\n            });\n            dispatch(this, 'dirty');\n        });\n        $graph.addEventListener('node-position-changed', (event) => {\n            const cEvent = event as CustomEvent<NodePositionChangedDetail>;\n            const queue = cEvent.detail.moveList.map((item) => {\n                return new BlockPositionAction({\n                    blockName: item.id,\n                    target: item.target,\n                    source: item.source,\n                });\n            });\n            if (queue.length === 1) {\n                this.actionQueue.exec(queue[0]);\n            } else if (queue.length > 1) {\n                this.actionQueue.exec(new ActionList({\n                    queue,\n                }));\n            }\n            dispatch(this, 'dirty', {\n                detail: {\n                    dirtyType: 'position-changed',\n                },\n            });\n        });\n        // //// ////\n        $graph.shadowRoot.addEventListener('dirty', (event) => {\n            const cEvent = event as CustomEvent<DirtyDetail>;\n            if (cEvent.detail && cEvent.detail.action) {\n                this.actionQueue.exec(cEvent.detail.action);\n            }\n            dispatch(this, 'dirty', {\n                detail: cEvent.detail,\n            });\n        });\n\n        $graph.addEventListener('mouseup', (event) => {\n            const info = graphMap.get(this.rootGraph!.type);\n            if (!info) {\n                return;\n            }\n            if ((event as MouseEvent).button === 2 && info.graph.event?.onGraphRightClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const graphPosition = $graph.convertCoordinate(event.offsetX, event.offsetY);\n                const customEvent = new GraphMouseEvent(nodes, lines, $graph, this);\n                customEvent.initPagePosition(event.pageX, event.pageY);\n                customEvent.initGraphPosition(graphPosition.x, graphPosition.y);\n                info.graph.event.onGraphRightClick(customEvent);\n            }\n        });\n\n        $graph.addEventListener('line-added', (event) => {\n            const customEment = event as CustomEvent<{line: LineData}>;\n            const $node = $graph.queryNodeElement(customEment.detail.line.output.node);\n            if ($node) {\n                // @ts-ignore\n                $node.onUpdate && $node.onUpdate();\n            }\n            dispatch(this, 'line-added', {\n                detail: customEment.detail,\n            });\n            dispatch(this, 'dirty');\n        });\n        $graph.addEventListener('line-removed', (event) => {\n            const customEment = event as CustomEvent<{line: LineData}>;\n            const $node = $graph.queryNodeElement(customEment.detail.line.output.node);\n            if ($node) {\n                // @ts-ignore\n                $node.onUpdate && $node.onUpdate();\n            }\n            dispatch(this, 'line-removed', {\n                detail: customEment.detail,\n            });\n            dispatch(this, 'dirty');\n        });\n        $graph.addEventListener('line-changed', (event) => {\n            const customElement = event as CustomEvent<{line: LineData}>;\n            dispatch(this, 'line-changed', {\n                detail: customElement.detail,\n            });\n            dispatch(this, 'dirty');\n        });\n\n        $graph.addEventListener('node-connected', (event) => {\n            const customElement = event as CustomEvent<{line: LineData}>;\n            this.startRecording();\n            this.addLine(customElement.detail.line);\n            setTimeout(() => {\n                this.stopRecording();\n            }, 200);\n        });\n\n        const $svg = $graph.shadowRoot.querySelector('#lines')!;\n        function searchG(htmlArray: (HTMLElement | SVGGElement)[]) {\n            const length = Math.min(htmlArray.length, 4);\n            for (let i = 0; i < length; i++) {\n                const $elem = htmlArray[i];\n                // 如果找到顶部的 document 元素的话，是没有 tagName 的\n                if ($elem.tagName && $elem.tagName.toLocaleLowerCase() === 'g') {\n                    return $elem as SVGGElement;\n                }\n            }\n        }\n        $svg.addEventListener('dblclick', (event) => {\n            // @ts-ignore\n            const $g = searchG(event.path);\n            if (!$g || !$g.hasAttribute('line-uuid')) {\n                return;\n            }\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            if (info.graph.event && info.graph.event.onLineDblClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $g.getAttribute('line-uuid') || '';\n                const line = lines[uuid] as LineData;\n                const event = new LineMouseEvent(nodes, lines, $g, line);\n                info.graph.event.onLineDblClick(event);\n            }\n        });\n        $svg.addEventListener('click', (event) => {\n            // @ts-ignore\n            const $g = searchG(event.path);\n            if (!$g || !$g.hasAttribute('line-uuid')) {\n                return;\n            }\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            if (info.graph.event && info.graph.event.onLineClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $g.getAttribute('line-uuid') || '';\n                const line = lines[uuid] as LineData;\n                const event = new LineMouseEvent(nodes, lines, $g, line);\n                info.graph.event.onLineClick(event);\n            }\n        });\n        $svg.addEventListener('mouseup', (event) => {\n            // @ts-ignore\n            const $g = searchG(event.path);\n            if (!$g || !$g.hasAttribute('line-uuid')) {\n                return;\n            }\n            if ((event as MouseEvent).button !== 2) {\n                return;\n            }\n\n            const type = this.paths[this.paths.length - 1].type;\n            const info = graphMap.get(type);\n            if (!info) {\n                return;\n            }\n            if (info.graph.event && info.graph.event.onLineRightClick) {\n                const nodes = $graph.getProperty('nodes') as { [uuid: string]: BlockData; };\n                const lines = $graph.getProperty('lines') as { [uuid: string]: LineData; };\n                const uuid = $g.getAttribute('line-uuid') || '';\n                const line = lines[uuid] as LineData;\n                const event = new LineMouseEvent(nodes, lines, $g, line);\n                info.graph.event.onLineRightClick(event);\n            }\n        });\n    }\n\n    private _updateGraph() {\n        clearDynamicEnum();\n        const graph = this.paths[this.paths.length - 1];\n        const $graph = this.shadowRoot!.querySelector('v-graph')! as GraphElement;\n        $graph.clear();\n        requestAnimationFrame(() => {\n            $graph.setAttribute('type', graph.type);\n            $graph.setProperty('lines', graph.lines);\n            $graph.setProperty('nodes', graph.nodes);\n            this._updateHeader();\n        });\n    }\n\n    public undo() {\n        this.actionQueue.undo();\n        dispatch(this, 'undo');\n    }\n\n    public redo() {\n        this.actionQueue.redo();\n        dispatch(this, 'redo');\n    }\n\n    public startRecording() {\n        this.actionQueue.startRecording();\n    }\n\n    public stopRecording() {\n        this.actionQueue.stopRecording();\n    }\n\n    public getPinElement(blockName: string, type: 'input' | 'output', index: number) {\n        const $block = this.$graph.shadowRoot.querySelector(`v-graph-node[node-uuid=${blockName}]`);\n        if (!$block) {\n            return;\n        }\n        const $pinList = $block.shadowRoot!.querySelectorAll(`.pin.in`);\n        const $pin = $pinList[index];\n        return $pin;\n    }\n\n    public getBlockElement(blockName: string) {\n        return this.$graph.shadowRoot.querySelector(`v-graph-node[node-uuid=${blockName}]`) as GraphNodeElement;\n    }\n\n    /// ---- 操作整个图\n\n    /**\n     * 将屏幕坐标转换成 Graph 内的坐标\n     * @param point\n     * @returns\n     */\n    convertCoordinate(point: { x: number, y: number }) {\n        point = this.$graph.convertCoordinate(point.x, point.y);\n        return point;\n    }\n\n    /**\n     * 设置编辑的根图\n     * @param graph\n     */\n    setRootGraph(graph: GraphData) {\n        this.rootGraph = graph;\n        this.paths = [graph];\n        this._updateGraph();\n    }\n\n    /**\n     * 获取正在编辑的根图\n     * @returns\n     */\n    getRootGraph(): GraphData {\n        return this.paths[0];\n    }\n\n    /**\n     * 传入一个字符串，反序列化成图数据\n     * @param content\n     * @returns\n     */\n    deserialize(content: string): GraphData {\n        const graphData = yaml.load(content) as GraphData;\n        return graphData;\n    }\n\n    /**\n     * 传入一个图数据，序列化成 yaml 字符串\n     * @param data\n     * @returns\n     */\n    serialize(data?: GraphData): string {\n        const str = yaml.dump(data || this.paths[0]);\n        // return JSON.stringify(this.paths[0]);\n        // outputFileSync('/Users/wangsijie/Project/Creator/cocos-editor/extension-repos/shader-graph/test.yaml', str);\n        return str;\n    }\n\n    /**\n     * 获取整个图现在的一些基础数据\n     * @returns\n     */\n    getGraphInfo() {\n        const offset = this.$graph.getProperty('offset');\n        const scale = this.$graph.getProperty('scale');\n        return {\n            offset, scale,\n        };\n    }\n\n    /**\n     * 设置整个图的一些基础数据\n     * @param info\n     */\n    setGraphInfo(info: { offset: { x: number, y: number, }, scale: number}) {\n        this.$graph.setProperty('offset', info.offset);\n        this.$graph.setProperty('scale', info.scale);\n    }\n\n    /**\n     * 恢复缩放比例\n     */\n    zoomToFit() {\n        this.$graph.data.setProperty('scale', 1);\n    }\n\n    /// ---- 操作当前图\n\n    /**\n     * 获取选中的 Block 列表\n     * @returns\n     */\n    getSelectedBlockList() {\n        return this.$graph.getSelectedNodeList();\n    }\n\n    /**\n     * 获取选中的 Line 列表\n     * @returns\n     */\n    getSelectedLineList() {\n        return this.$graph.getSelectedLineList();\n    }\n\n    /**\n     * 设置当前正在编辑的图数据\n     * @param graph\n     * @returns\n     */\n    setCurrentGraph(graph: GraphData) {\n        if (this.paths.length <= 1) {\n            this.setRootGraph(graph);\n            return;\n        }\n        this.paths[this.paths.length - 1] = graph;\n        this._updateGraph();\n    }\n\n    /**\n     * 获取正在编辑的图数据\n     * @returns\n     */\n    getCurrentGraph() {\n        return this.paths[this.paths.length - 1];\n    }\n\n    /**\n     * 在当前正在操作的图数据里增加一个 Block\n     * @param block\n     * @param id\n     */\n    addBlock(block: BlockData, id?: string) {\n        this.actionQueue.exec(new AddBlockAction({ block, id }));\n    }\n\n    /**\n     * 在当前正在操作的图数据里删除一个节点\n     * @param id\n     */\n    removeBlock(id: string) {\n        const queue: Action[] = [];\n        // remove line\n        const lines = this.$graph.getProperty('lines') as { [uuid: string]: LineData; };\n        for (const key in lines) {\n            const line = lines[key] as LineData;\n            if (line.input.node === id || line.output.node === id) {\n                queue.push(new RemoveLineAction({ id: key }));\n            }\n        }\n        queue.push(new RemoveBlockAction({ id }));\n        this.actionQueue.exec(new ActionList({\n            queue,\n        }));\n    }\n\n    /**\n     * 在当前正在操作的图数据里增加一个连线\n     * @param line\n     * @param id\n     */\n    addLine(line: LineData, id?: string) {\n        this.actionQueue.exec(new AddLineAction({ line, id }));\n    }\n\n    /**\n     * 在当前正在操作的图数据里删除一个连线\n     * @param id\n     */\n    removeLine(id: string) {\n        this.actionQueue.exec(new RemoveLineAction({ id }));\n    }\n\n    /**\n     * 进入当前图的子图\n     * @param id\n     */\n    enterSubGraph(id: string) {\n        const graph = this.paths[this.paths.length - 1];\n        const subGraph = graph.graphs[id];\n        if (subGraph) {\n            this.paths.push(subGraph);\n            this._updateGraph();\n        }\n        dispatch(this, 'enter-graph', {\n            detail: {\n                id: id,\n            },\n        });\n    }\n\n    /**\n     * 在当前编辑的图里增加一个子图\n     * @param type\n     * @param id\n     * @returns\n     */\n    addSubGraph(type: string, id: string) {\n        const info = this.paths[this.paths.length - 1];\n        // const uuid = generateUUID();\n        info.graphs[id] = {\n            type,\n            name: type,\n            nodes: {},\n            lines: {},\n            graphs: {},\n        } as GraphData;\n\n        return info.graphs[id];\n    }\n\n    /**\n     * 在当前编辑的图里，删除一个子图\n     * @param id\n     */\n    removeSubGraph(id: string) {\n        const info = this.paths[this.paths.length - 1];\n        delete info.graphs[id];\n    }\n}\n\nif (!window.customElements.get('ui-graph-forge')) {\n    window.customElements.define('ui-graph-forge', HTMLGraphForgeElement);\n}\n"]}