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

419 lines
45 KiB
JavaScript

'use strict';
Object.defineProperty(exports, "__esModule", { value: true });
exports.replaceDeclareBlock = exports.unDeclareBlock = exports.declareBlock = exports.removeDeclareBlock = exports.getDeclareBlock = exports.hasDeclareBlock = exports.generateBlockOption = exports.blockMap = void 0;
const ui_graph_1 = require("@itharbors/ui-graph");
const utils_1 = require("./utils");
const pin_1 = require("./pin");
// import { unregisterNode } from '@itharbors/ui-graph/dist/manager';
exports.blockMap = new Map();
/**
* Block 元素更新的一些工具方法
* 一般是传入元素 + 数据,更新元素内的一些 HTML 对象
*/
const BlockElementUtils = {
/**
* 更新 Block 元素的 title
* @param elem
* @param blockDesc
* @param details
*/
updateTitle(elem, blockDesc, details) {
const title = blockDesc.title || details.title || 'Unknown';
elem.shadowRoot.querySelector(`.title ui-label`).innerHTML = title;
},
/**
* 更新 Block 元素的 icon
* @param elem
* @param blockDesc
* @returns
*/
updateIcon(elem, blockDesc) {
const feature = blockDesc.feature || {};
const icon = feature.icon;
if (!icon) {
return;
}
const $icon = elem.shadowRoot.querySelector(`.title ui-icon`);
$icon.removeAttribute('hidden');
$icon.setAttribute('value', icon);
},
/**
* 更新 Block 元素是否可进入的图标显示状态
* @param elem
* @param blockDesc
*/
updateCollapsed(elem, blockDesc) {
const feature = blockDesc.feature || {};
const isCollapsed = feature.isCollapsedBlock;
const $svg = elem.shadowRoot.querySelector(`.title svg`);
if (isCollapsed) {
$svg.removeAttribute('hidden');
}
else {
$svg.setAttribute('hidden', '');
}
},
/**
* 更新元素展开折叠的状态
* @param elem
* @param blockDesc
* @param details
*/
updateExpand(elem, blockDesc, details) {
if (blockDesc.inputPins.length > 0 || blockDesc.outputPins.length > 0) {
elem.setAttribute('expand', '');
}
else {
elem.removeAttribute('expand');
}
},
};
/**
* 创建一个专用的节点渲染对象
*
* @returns
* @param blockDesc
*/
function generateBlockOption(blockDesc) {
blockDesc.style = blockDesc.style || {};
blockDesc.feature = blockDesc.feature || {};
const showQuickConnectPoint = !!blockDesc.feature.showQuickConnectPoint;
return {
template: /*html*/ `
<section class="wrapper">
<header class="title">
<div>
<ui-icon hidden></ui-icon>
<ui-label></ui-label>
<svg hidden viewBox="0 0 16 16" xmlns="http://www.w3.org/2000/svg"><path d="M1 13L8 3L15 13H1Z"></path></svg>
</div>
${showQuickConnectPoint ? '<div hidden class="quick-connect" name="t"></div' : ''}
</header>
<section class="content"></section>
</section>
`,
style: `${STYLE.host(blockDesc)}${STYLE.header(blockDesc)}${STYLE.pin(blockDesc)}`,
/**
* 初始化的时候设置一些事件和 HTML
* @param this
* @param details
*/
onInit(details) {
// 设置 title 可拖拽
const $title = this.shadowRoot.querySelector('header.title');
$title.addEventListener('mousedown', (event) => {
event.stopPropagation();
event.preventDefault();
if (!this.hasAttribute('selected')) {
if (!event.metaKey && !event.ctrlKey) {
this.clearOtherSelected();
}
this.select({
clearLines: false,
clearNodes: false,
});
}
this.startMove();
});
// 绑定快速连接点的事件
const $param = this.shadowRoot.querySelector(`.quick-connect`);
$param && $param.addEventListener('mousedown', () => {
this.startConnect('straight');
});
// 绑定元素点击开始连接的事件
if (blockDesc.inputPins.length === 0 && blockDesc.outputPins.length === 0) {
this.addEventListener('mousedown', (event) => {
if (event.button === 0 && this.hasConnect()) {
event.stopPropagation();
event.preventDefault();
this.startConnect('straight');
}
}, true);
}
this.addEventListener('dblclick', (event) => {
// event.stopPropagation();
// event.preventDefault();
const customEvent = new CustomEvent('block-dblclick', {
bubbles: true,
cancelable: true,
detail: {
pageX: event.pageX,
pageY: event.pageY,
offsetX: event.offsetX,
offsetY: event.offsetY,
},
});
this.dispatchEvent(customEvent);
});
this.addEventListener('click', (event) => {
const custom = new CustomEvent('block-click', {
bubbles: true,
cancelable: true,
detail: {},
});
this.dispatchEvent(custom);
});
this.addEventListener('mouseup', (event) => {
if (event.button !== 2) {
return;
}
const custom = new CustomEvent('block-right-click', {
bubbles: true,
cancelable: true,
detail: {},
});
this.dispatchEvent(custom);
});
this.data.addPropertyListener('selected', (selected, legacySelected) => {
if (selected === legacySelected) {
return;
}
if (selected) {
const custom = new CustomEvent('block-selected', {
bubbles: true,
cancelable: true,
detail: {},
});
this.dispatchEvent(custom);
}
else {
const custom = new CustomEvent('block-unselected', {
bubbles: true,
cancelable: true,
detail: {},
});
this.dispatchEvent(custom);
}
});
},
onUpdate(details) {
// 更新 title
BlockElementUtils.updateTitle(this, blockDesc, details);
// 更新 icon
BlockElementUtils.updateIcon(this, blockDesc);
// 更新折叠图标
BlockElementUtils.updateCollapsed(this, blockDesc);
// 更新折叠状态
BlockElementUtils.updateExpand(this, blockDesc, details);
// 数据更新后,更新对应的资源
// this.data.addPropertyListener('details', (details) => {
// updateHTML(details.label);
// });
// 生成针脚
const $content = this.shadowRoot.querySelector('.content');
$content.innerHTML = '';
if (blockDesc.createDynamicOutputPins) {
const outputList = blockDesc.createDynamicOutputPins(blockDesc, details);
outputList.forEach((pin, index) => {
return $content?.appendChild((0, pin_1.generateOutputPinHTML)(pin, details.outputPins[index]));
});
}
else {
blockDesc.outputPins.forEach((pin, index) => {
return $content?.appendChild((0, pin_1.generateOutputPinHTML)(pin, details.outputPins[index]));
});
}
const $graph = this.getRootNode().host;
const uuid = this.getAttribute('node-uuid');
if (blockDesc.createDynamicInputPins) {
const inputList = blockDesc.createDynamicInputPins(blockDesc, details);
inputList.forEach((pin, index) => {
return $content?.appendChild((0, pin_1.generateInputPinHTML)(pin, details.inputPins, index, uuid, $graph.getProperty('lines')));
});
}
else {
blockDesc.inputPins.forEach((pin, index) => {
return $content?.appendChild((0, pin_1.generateInputPinHTML)(pin, details.inputPins, index, uuid, $graph.getProperty('lines')));
});
}
// 绑定参数连接点的事件
const $paramList = this.shadowRoot.querySelectorAll(`v-graph-node-param`);
Array.prototype.forEach.call($paramList, ($param) => {
$param.addEventListener('mousedown', (event) => {
event.stopPropagation();
event.preventDefault();
const name = $param.getAttribute('name');
if (!name) {
return;
}
const paramDirection = $param.getAttribute('direction');
if (paramDirection !== 'input' && paramDirection !== 'output') {
return;
}
this.startConnect('curve', name, paramDirection);
});
});
},
};
}
exports.generateBlockOption = generateBlockOption;
const STYLE = {
host(block) {
const config = block.style || {};
return /*css*/ `
:host *[hidden] {
display: none;
}
:host {
--font-color: ${config.fontColor || '#ccc'};
--font-color-hover: ${config.fontHoverColor || config.fontColor || '#ccc'};
--border-color: ${config.borderColor || 'white'};
--border-color-hover: ${config.borderHoverColor || config.borderColor || 'white'};
--shadow-color: ${config.shadowColor || '#ccc'};
--shadow-color-hover: ${config.shadowHoverColor || config.shadowColor || '#ccc'};
--background-color: ${config.backgroundColor || '#2b2b2bcc'};
--border-radius: 2px;
--header-height: 24px;
--header-background: ${config.headerColor || '#2b2b2bcc'};
--pin-height: 24px;
width: 200px;
color: var(--font-color);
cursor: default;
}
:host > section.wrapper {
margin: 10px;
}
:host(:hover) > section.wrapper, :host([selected]) > section.wrapper {
border-color: var(--border-color-hover);
color: var(--font-color-hover);
box-shadow: 0px 0px 7px 2px var(--shadow-color-hover);
}
section.wrapper {
position: relative;
border-radius: var(--border-radius);
background: var(--background-color);
box-shadow: 0px 0px 7px 2px none;
}
:host(:hover) > section.wrapper::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
z-index: 1;
border-radius: var(--border-radius);
box-shadow: 0px 0px 0px 1px var(--shadow-color-hover) inset;
pointer-events: none;
}
`;
},
header(block) {
const config = block.style || {};
return /*css*/ `
header.title {
line-height: var(--header-height);
text-align: center;
border-radius: var(--border-radius);
${config.secondaryColor ? `background: ${config.secondaryColor}; padding-left: 6px;` : ''}
}
header.title > div {
padding: 0 10px;
height: 24px;
border-radius: var(--border-radius);
display: flex;
background: var(--header-background);
}
:host([expand]) header.title > div {
border-radius: var(--border-radius) var(--border-radius) 0 0;
}
header.title > div > ui-label {
display: block;
padding: 0 10px;
}
header.title > div > ui-icon {
display: block;
}
header.title > div > svg {
fill: white;
display: block;
width: 10px;
transform: rotate(90deg);
}
header.title > .quick-connect {
display: block;
padding: 0;
width: 12px;
height: 12px;
border-radius: 6px;
background: white;
position: absolute;
right: -6px;
top: 50%;
margin-top: -6px;
opacity: 0;
transition: opacity 0.3s;
}
:host(:hover) header.title > .quick-connect {
display: block;
opacity: 1;
}
`;
},
pin: pin_1.generateStyle,
};
function hasDeclareBlock(type) {
return exports.blockMap.has(type);
}
exports.hasDeclareBlock = hasDeclareBlock;
function getDeclareBlock(type) {
return exports.blockMap.get(type);
}
exports.getDeclareBlock = getDeclareBlock;
function removeDeclareBlock(type) {
const graph = '*';
const unknownOption = (0, ui_graph_1.queryNode)(graph, 'unknown');
(0, ui_graph_1.registerNode)(graph, type, unknownOption);
exports.blockMap.delete(type);
}
exports.removeDeclareBlock = removeDeclareBlock;
// @ts-ignore
window.removeDeclareBlock = removeDeclareBlock;
/**
* 注册一个 block 类型
* @param block
* @returns
*/
function declareBlock(block) {
const graph = '*';
// if (blockMap.has(block.type)) {
// console.warn(`Cannot declare duplicate block types: ${block.type}`);
// return;
// }
// 合并 extend 数据
if (block.extend) {
const extend = exports.blockMap.get(block.extend);
if (!extend) {
console.warn(`Inheritance data not found: ${block.extend}`);
}
else {
(0, utils_1.completeBlockTarget)(block, extend);
}
}
const options = generateBlockOption(block);
(0, ui_graph_1.registerNode)(graph, block.type, options);
exports.blockMap.set(block.type, block);
}
exports.declareBlock = declareBlock;
function unDeclareBlock(type) {
const graph = '*';
// TODO 需要删除底层 block 节点
// unregisterNode(graph, type);
exports.blockMap.delete(type);
}
exports.unDeclareBlock = unDeclareBlock;
function replaceDeclareBlock(searchType, block) {
if (exports.blockMap.has(searchType)) {
unDeclareBlock(searchType);
}
declareBlock(block);
}
exports.replaceDeclareBlock = replaceDeclareBlock;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"block.js","sourceRoot":"","sources":["../../src/block-forge/block.ts"],"names":[],"mappings":"AAAA,YAAY,CAAC;;;AAIb,kDAA8D;AAE9D,mCAA8C;AAE9C,+BAAuG;AACvG,qEAAqE;AAExD,QAAA,QAAQ,GAAmC,IAAI,GAAG,EAAE,CAAC;AAElE;;;GAGG;AACH,MAAM,iBAAiB,GAAG;IACtB;;;;;OAKG;IACH,WAAW,CAAC,IAAsB,EAAE,SAA4B,EAAE,OAA+B;QAC7F,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,IAAI,OAAO,CAAC,KAAK,IAAI,SAAS,CAAC;QAC5D,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,iBAAiB,CAAE,CAAC,SAAS,GAAG,KAAK,CAAC;IACxE,CAAC;IAED;;;;;OAKG;IACH,UAAU,CAAC,IAAsB,EAAE,SAA4B;QAC3D,MAAM,OAAO,GAAG,SAAS,CAAC,OAAO,IAAI,EAAE,CAAC;QACxC,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;QAC1B,IAAI,CAAC,IAAI,EAAE;YACP,OAAO;SACV;QACD,MAAM,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,gBAAgB,CAAE,CAAC;QAC/D,KAAK,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;QAChC,KAAK,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACtC,CAAC;IAED;;;;OAIG;IACH,eAAe,CAAC,IAAsB,EAAE,SAA4B;QAChE,MAAM,OAAO,GAAG,SAAS,CAAC,OAAO,IAAI,EAAE,CAAC;QACxC,MAAM,WAAW,GAAG,OAAO,CAAC,gBAAgB,CAAC;QAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,YAAY,CAAE,CAAC;QAC1D,IAAI,WAAW,EAAE;YACb,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;SAClC;aAAM;YACH,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;SACnC;IACL,CAAC;IAED;;;;;OAKG;IACH,YAAY,CAAC,IAAsB,EAAE,SAA4B,EAAE,OAA+B;QAC9F,IAAI,SAAS,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,IAAI,SAAS,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE;YACnE,IAAI,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;SACnC;aAAM;YACH,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;SAClC;IACL,CAAC;CACJ,CAAC;AAEF;;;;;GAKG;AACH,SAAgB,mBAAmB,CAAC,SAA4B;IAC5D,SAAS,CAAC,KAAK,GAAG,SAAS,CAAC,KAAK,IAAI,EAAE,CAAC;IACxC,SAAS,CAAC,OAAO,GAAG,SAAS,CAAC,OAAO,IAAI,EAAE,CAAC;IAE5C,MAAM,qBAAqB,GAAG,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,qBAAqB,CAAC;IAExE,OAAO;QACH,QAAQ,EAAE,QAAQ,CAAA;;;;;;;;UAQhB,qBAAqB,CAAC,CAAC,CAAC,kDAAkD,CAAC,CAAC,CAAC,EAAE;;;;SAIhF;QAED,KAAK,EAAE,GAAG,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC,GAAG,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE;QAElF;;;;WAIG;QACH,MAAM,CAAyB,OAA+B;YAC1D,eAAe;YACf,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,cAAc,CAAiB,CAAC;YAC7E,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,EAAE;gBAC3C,KAAK,CAAC,eAAe,EAAE,CAAC;gBACxB,KAAK,CAAC,cAAc,EAAE,CAAC;gBACvB,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,EAAE;oBAChC,IAAI,CAAE,KAAoB,CAAC,OAAO,IAAI,CAAE,KAAoB,CAAC,OAAO,EAAE;wBAClE,IAAI,CAAC,kBAAkB,EAAE,CAAC;qBAC7B;oBACD,IAAI,CAAC,MAAM,CAAC;wBACR,UAAU,EAAE,KAAK;wBACjB,UAAU,EAAE,KAAK;qBACpB,CAAC,CAAC;iBACN;gBACD,IAAI,CAAC,SAAS,EAAE,CAAC;YACrB,CAAC,CAAC,CAAC;YAEH,aAAa;YACb,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,gBAAgB,CAAE,CAAC;YAChE,MAAM,IAAI,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,GAAG,EAAE;gBAChD,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;YAClC,CAAC,CAAC,CAAC;YAEH,gBAAgB;YAChB,IAAI,SAAS,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,IAAI,SAAS,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE;gBACvE,IAAI,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC,KAAK,EAAE,EAAE;oBACzC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,IAAI,IAAI,CAAC,UAAU,EAAE,EAAE;wBACzC,KAAK,CAAC,eAAe,EAAE,CAAC;wBACxB,KAAK,CAAC,cAAc,EAAE,CAAC;wBACvB,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,CAAC;qBACjC;gBACL,CAAC,EAAE,IAAI,CAAC,CAAC;aACZ;YAED,IAAI,CAAC,gBAAgB,CAAC,UAAU,EAAE,CAAC,KAAK,EAAE,EAAE;gBACxC,2BAA2B;gBAC3B,0BAA0B;gBAE1B,MAAM,WAAW,GAAG,IAAI,WAAW,CAAC,gBAAgB,EAAE;oBAClD,OAAO,EAAE,IAAI;oBACb,UAAU,EAAE,IAAI;oBAChB,MAAM,EAAE;wBACJ,KAAK,EAAE,KAAK,CAAC,KAAK;wBAClB,KAAK,EAAE,KAAK,CAAC,KAAK;wBAClB,OAAO,EAAE,KAAK,CAAC,OAAO;wBACtB,OAAO,EAAE,KAAK,CAAC,OAAO;qBACzB;iBACJ,CAAC,CAAC;gBACH,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC;YACpC,CAAC,CAAC,CAAC;YACH,IAAI,CAAC,gBAAgB,CAAC,OAAO,EAAE,CAAC,KAAK,EAAE,EAAE;gBACrC,MAAM,MAAM,GAAG,IAAI,WAAW,CAAC,aAAa,EAAE;oBAC1C,OAAO,EAAE,IAAI;oBACb,UAAU,EAAE,IAAI;oBAChB,MAAM,EAAE,EAAE;iBACb,CAAC,CAAC;gBACH,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;YAC/B,CAAC,CAAC,CAAC;YACH,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,CAAC,KAAK,EAAE,EAAE;gBACvC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE;oBACpB,OAAO;iBACV;gBACD,MAAM,MAAM,GAAG,IAAI,WAAW,CAAC,mBAAmB,EAAE;oBAChD,OAAO,EAAE,IAAI;oBACb,UAAU,EAAE,IAAI;oBAChB,MAAM,EAAE,EACP;iBACJ,CAAC,CAAC;gBACH,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;YAC/B,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,UAAU,EAAE,CAAC,QAAQ,EAAE,cAAc,EAAE,EAAE;gBACnE,IAAI,QAAQ,KAAK,cAAc,EAAE;oBAC7B,OAAO;iBACV;gBACD,IAAI,QAAQ,EAAE;oBACV,MAAM,MAAM,GAAG,IAAI,WAAW,CAAC,gBAAgB,EAAE;wBAC7C,OAAO,EAAE,IAAI;wBACb,UAAU,EAAE,IAAI;wBAChB,MAAM,EAAE,EAAE;qBACb,CAAC,CAAC;oBACH,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;iBAC9B;qBAAM;oBACH,MAAM,MAAM,GAAG,IAAI,WAAW,CAAC,kBAAkB,EAAE;wBAC/C,OAAO,EAAE,IAAI;wBACb,UAAU,EAAE,IAAI;wBAChB,MAAM,EAAE,EAAE;qBACb,CAAC,CAAC;oBACH,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;iBAC9B;YACL,CAAC,CAAC,CAAC;QACP,CAAC;QAED,QAAQ,CAAyB,OAA+B;YAC5D,WAAW;YACX,iBAAiB,CAAC,WAAW,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;YAExD,UAAU;YACV,iBAAiB,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;YAE9C,SAAS;YACT,iBAAiB,CAAC,eAAe,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;YAEnD,SAAS;YACT,iBAAiB,CAAC,YAAY,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;YAEzD,gBAAgB;YAChB,0DAA0D;YAC1D,iCAAiC;YACjC,MAAM;YAEN,OAAO;YACP,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,UAAU,CAAE,CAAC;YAC5D,QAAQ,CAAC,SAAS,GAAG,EAAE,CAAC;YACxB,IAAI,SAAS,CAAC,uBAAuB,EAAE;gBACnC,MAAM,UAAU,GAAG,SAAS,CAAC,uBAAuB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;gBACzE,UAAU,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;oBAC9B,OAAO,QAAQ,EAAE,WAAW,CAAC,IAAA,2BAAqB,EAAC,GAAG,EAAE,OAAO,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBACxF,CAAC,CAAC,CAAC;aACN;iBAAM;gBACH,SAAS,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;oBACxC,OAAO,QAAQ,EAAE,WAAW,CAAC,IAAA,2BAAqB,EAAC,GAAG,EAAE,OAAO,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBACxF,CAAC,CAAC,CAAC;aACN;YACD,MAAM,MAAM,GAAI,IAAI,CAAC,WAAW,EAAU,CAAC,IAAoB,CAAC;YAChE,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,CAAC,WAAW,CAAE,CAAC;YAC7C,IAAI,SAAS,CAAC,sBAAsB,EAAE;gBAClC,MAAM,SAAS,GAAG,SAAS,CAAC,sBAAsB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;gBACvE,SAAS,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;oBAC7B,OAAO,QAAQ,EAAE,WAAW,CAAC,IAAA,0BAAoB,EAAC,GAAG,EAAE,OAAO,CAAC,SAAS,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACzH,CAAC,CAAC,CAAC;aACN;iBAAM;gBACH,SAAS,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;oBACvC,OAAO,QAAQ,EAAE,WAAW,CAAC,IAAA,0BAAoB,EAAC,GAAG,EAAE,OAAO,CAAC,SAAS,EAAE,KAAK,EAAE,IAAI,EAAE,MAAM,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACzH,CAAC,CAAC,CAAC;aACN;YAED,aAAa;YACb,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,oBAAoB,CAAC,CAAC;YAC1E,KAAK,CAAC,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,MAAM,EAAE,EAAE;gBAChD,MAAM,CAAC,gBAAgB,CAAC,WAAW,EAAE,CAAC,KAAiB,EAAE,EAAE;oBACvD,KAAK,CAAC,eAAe,EAAE,CAAC;oBACxB,KAAK,CAAC,cAAc,EAAE,CAAC;oBAEvB,MAAM,IAAI,GAAG,MAAM,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;oBACzC,IAAI,CAAC,IAAI,EAAE;wBACP,OAAO;qBACV;oBACD,MAAM,cAAc,GAAG,MAAM,CAAC,YAAY,CAAC,WAAW,CAAC,CAAC;oBACxD,IAAI,cAAc,KAAK,OAAO,IAAI,cAAc,KAAK,QAAQ,EAAE;wBAC3D,OAAO;qBACV;oBACD,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,EAAE,cAAc,CAAC,CAAC;gBACrD,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;QACP,CAAC;KACJ,CAAC;AACN,CAAC;AA1LD,kDA0LC;AAED,MAAM,KAAK,GAAG;IACV,IAAI,CAAC,KAAwB;QACzB,MAAM,MAAM,GAAG,KAAK,CAAC,KAAK,IAAI,EAAE,CAAC;QACjC,OAAO,OAAO,CAAA;;;;;oBAKF,MAAM,CAAC,SAAS,IAAI,MAAM;0BACpB,MAAM,CAAC,cAAc,IAAI,MAAM,CAAC,SAAS,IAAI,MAAM;sBACvD,MAAM,CAAC,WAAW,IAAI,OAAO;4BACvB,MAAM,CAAC,gBAAgB,IAAI,MAAM,CAAC,WAAW,IAAI,OAAO;sBAC9D,MAAM,CAAC,WAAW,IAAI,MAAM;4BACtB,MAAM,CAAC,gBAAgB,IAAI,MAAM,CAAC,WAAW,IAAI,MAAM;0BACzD,MAAM,CAAC,eAAe,IAAI,WAAW;;;;2BAIpC,MAAM,CAAC,WAAW,IAAI,WAAW;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAoC3D,CAAC;IACE,CAAC;IACD,MAAM,CAAC,KAAwB;QAC3B,MAAM,MAAM,GAAG,KAAK,CAAC,KAAK,IAAI,EAAE,CAAC;QACjC,OAAO,OAAO,CAAA;;;;;;;MAOhB,MAAM,CAAC,cAAc,CAAC,CAAC,CAAC,eAAe,MAAM,CAAC,cAAc,sBAAsB,CAAC,CAAC,CAAC,EAAE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;SA2CpF,CAAC;IACN,CAAC;IACD,GAAG,EAAE,mBAAgB;CACxB,CAAC;AAEF,SAAgB,eAAe,CAAC,IAAY;IACxC,OAAO,gBAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;AAC9B,CAAC;AAFD,0CAEC;AAED,SAAgB,eAAe,CAAC,IAAY;IACxC,OAAO,gBAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;AAC9B,CAAC;AAFD,0CAEC;AAED,SAAgB,kBAAkB,CAAC,IAAY;IAC3C,MAAM,KAAK,GAAG,GAAG,CAAC;IAClB,MAAM,aAAa,GAAG,IAAA,oBAAS,EAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IAClD,IAAA,uBAAY,EAAC,KAAK,EAAE,IAAI,EAAE,aAAa,CAAC,CAAC;IACzC,gBAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AALD,gDAKC;AAED,aAAa;AACb,MAAM,CAAC,kBAAkB,GAAG,kBAAkB,CAAC;AAE/C;;;;GAIG;AACH,SAAgB,YAAY,CAAC,KAAwB;IACjD,MAAM,KAAK,GAAG,GAAG,CAAC;IAClB,kCAAkC;IAClC,2EAA2E;IAC3E,cAAc;IACd,IAAI;IAEJ,eAAe;IACf,IAAI,KAAK,CAAC,MAAM,EAAE;QACd,MAAM,MAAM,GAAG,gBAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAC1C,IAAI,CAAC,MAAM,EAAE;YACT,OAAO,CAAC,IAAI,CAAC,+BAA+B,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC;SAC/D;aAAM;YACH,IAAA,2BAAmB,EAAC,KAAK,EAAE,MAAM,CAAC,CAAC;SACtC;KACJ;IAED,MAAM,OAAO,GAAG,mBAAmB,CAAC,KAAK,CAAC,CAAC;IAC3C,IAAA,uBAAY,EAAC,KAAK,EAAE,KAAK,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;IACzC,gBAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;AACpC,CAAC;AApBD,oCAoBC;AAED,SAAgB,cAAc,CAAC,IAAY;IACvC,MAAM,KAAK,GAAG,GAAG,CAAC;IAClB,uBAAuB;IACvB,+BAA+B;IAC/B,gBAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;AAC1B,CAAC;AALD,wCAKC;AAED,SAAgB,mBAAmB,CAAC,UAAkB,EAAE,KAAwB;IAC5E,IAAI,gBAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,EAAE;QAC1B,cAAc,CAAC,UAAU,CAAC,CAAC;KAC9B;IACD,YAAY,CAAC,KAAK,CAAC,CAAC;AACxB,CAAC;AALD,kDAKC","sourcesContent":["'use strict';\n\nimport type { GraphNodeElement } from '@itharbors/ui-graph/dist/element/graph-node';\nimport type { GraphElement } from '@itharbors/ui-graph/dist/element/graph';\nimport { registerNode, queryNode } from '@itharbors/ui-graph';\nimport type { IPinDescription, IBlockDescription } from './interface';\nimport { completeBlockTarget } from './utils';\n\nimport { generateOutputPinHTML, generateInputPinHTML, generateStyle as generatePinStyle } from './pin';\n// import { unregisterNode } from '@itharbors/ui-graph/dist/manager';\n\nexport const blockMap: Map<string, IBlockDescription> = new Map();\n\n/**\n * Block 元素更新的一些工具方法\n * 一般是传入元素 + 数据，更新元素内的一些 HTML 对象\n */\nconst BlockElementUtils = {\n    /**\n     * 更新 Block 元素的 title\n     * @param elem\n     * @param blockDesc\n     * @param details\n     */\n    updateTitle(elem: GraphNodeElement, blockDesc: IBlockDescription, details: { [key: string]: any }) {\n        const title = blockDesc.title || details.title || 'Unknown';\n        elem.shadowRoot.querySelector(`.title ui-label`)!.innerHTML = title;\n    },\n\n    /**\n     * 更新 Block 元素的 icon\n     * @param elem\n     * @param blockDesc\n     * @returns\n     */\n    updateIcon(elem: GraphNodeElement, blockDesc: IBlockDescription) {\n        const feature = blockDesc.feature || {};\n        const icon = feature.icon;\n        if (!icon) {\n            return;\n        }\n        const $icon = elem.shadowRoot.querySelector(`.title ui-icon`)!;\n        $icon.removeAttribute('hidden');\n        $icon.setAttribute('value', icon);\n    },\n\n    /**\n     * 更新 Block 元素是否可进入的图标显示状态\n     * @param elem\n     * @param blockDesc\n     */\n    updateCollapsed(elem: GraphNodeElement, blockDesc: IBlockDescription) {\n        const feature = blockDesc.feature || {};\n        const isCollapsed = feature.isCollapsedBlock;\n        const $svg = elem.shadowRoot.querySelector(`.title svg`)!;\n        if (isCollapsed) {\n            $svg.removeAttribute('hidden');\n        } else {\n            $svg.setAttribute('hidden', '');\n        }\n    },\n\n    /**\n     * 更新元素展开折叠的状态\n     * @param elem\n     * @param blockDesc\n     * @param details\n     */\n    updateExpand(elem: GraphNodeElement, blockDesc: IBlockDescription, details: { [key: string]: any }) {\n        if (blockDesc.inputPins.length > 0 || blockDesc.outputPins.length > 0) {\n            elem.setAttribute('expand', '');\n        } else {\n            elem.removeAttribute('expand');\n        }\n    },\n};\n\n/**\n * 创建一个专用的节点渲染对象\n *\n * @returns\n * @param blockDesc\n */\nexport function generateBlockOption(blockDesc: IBlockDescription) {\n    blockDesc.style = blockDesc.style || {};\n    blockDesc.feature = blockDesc.feature || {};\n\n    const showQuickConnectPoint = !!blockDesc.feature.showQuickConnectPoint;\n\n    return {\n        template: /*html*/`\n<section class=\"wrapper\">\n    <header class=\"title\">\n        <div>\n            <ui-icon hidden></ui-icon>\n            <ui-label></ui-label>\n            <svg hidden viewBox=\"0 0 16 16\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"M1 13L8 3L15 13H1Z\"></path></svg>\n        </div>\n        ${showQuickConnectPoint ? '<div hidden class=\"quick-connect\" name=\"t\"></div' : ''}\n    </header>\n    <section class=\"content\"></section>\n</section>\n        `,\n\n        style: `${STYLE.host(blockDesc)}${STYLE.header(blockDesc)}${STYLE.pin(blockDesc)}`,\n\n        /**\n         * 初始化的时候设置一些事件和 HTML\n         * @param this\n         * @param details\n         */\n        onInit(this: GraphNodeElement, details: { [key: string]: any }) {\n            // 设置 title 可拖拽\n            const $title = this.shadowRoot.querySelector('header.title')! as HTMLElement;\n            $title.addEventListener('mousedown', (event) => {\n                event.stopPropagation();\n                event.preventDefault();\n                if (!this.hasAttribute('selected')) {\n                    if (!(event as MouseEvent).metaKey && !(event as MouseEvent).ctrlKey) {\n                        this.clearOtherSelected();\n                    }\n                    this.select({\n                        clearLines: false,\n                        clearNodes: false,\n                    });\n                }\n                this.startMove();\n            });\n\n            // 绑定快速连接点的事件\n            const $param = this.shadowRoot.querySelector(`.quick-connect`)!;\n            $param && $param.addEventListener('mousedown', () => {\n                this.startConnect('straight');\n            });\n\n            // 绑定元素点击开始连接的事件\n            if (blockDesc.inputPins.length === 0 && blockDesc.outputPins.length === 0) {\n                this.addEventListener('mousedown', (event) => {\n                    if (event.button === 0 && this.hasConnect()) {\n                        event.stopPropagation();\n                        event.preventDefault();\n                        this.startConnect('straight');\n                    }\n                }, true);\n            }\n\n            this.addEventListener('dblclick', (event) => {\n                // event.stopPropagation();\n                // event.preventDefault();\n\n                const customEvent = new CustomEvent('block-dblclick', {\n                    bubbles: true,\n                    cancelable: true,\n                    detail: {\n                        pageX: event.pageX,\n                        pageY: event.pageY,\n                        offsetX: event.offsetX,\n                        offsetY: event.offsetY,\n                    },\n                });\n                this.dispatchEvent(customEvent);\n            });\n            this.addEventListener('click', (event) => {\n                const custom = new CustomEvent('block-click', {\n                    bubbles: true,\n                    cancelable: true,\n                    detail: {},\n                });\n                this.dispatchEvent(custom);\n            });\n            this.addEventListener('mouseup', (event) => {\n                if (event.button !== 2) {\n                    return;\n                }\n                const custom = new CustomEvent('block-right-click', {\n                    bubbles: true,\n                    cancelable: true,\n                    detail: {\n                    },\n                });\n                this.dispatchEvent(custom);\n            });\n\n            this.data.addPropertyListener('selected', (selected, legacySelected) => {\n                if (selected === legacySelected) {\n                    return;\n                }\n                if (selected) {\n                    const custom = new CustomEvent('block-selected', {\n                        bubbles: true,\n                        cancelable: true,\n                        detail: {},\n                    });\n                    this.dispatchEvent(custom);\n                } else {\n                    const custom = new CustomEvent('block-unselected', {\n                        bubbles: true,\n                        cancelable: true,\n                        detail: {},\n                    });\n                    this.dispatchEvent(custom);\n                }\n            });\n        },\n\n        onUpdate(this: GraphNodeElement, details: { [key: string]: any }) {\n            // 更新 title\n            BlockElementUtils.updateTitle(this, blockDesc, details);\n\n            // 更新 icon\n            BlockElementUtils.updateIcon(this, blockDesc);\n\n            // 更新折叠图标\n            BlockElementUtils.updateCollapsed(this, blockDesc);\n\n            // 更新折叠状态\n            BlockElementUtils.updateExpand(this, blockDesc, details);\n\n            // 数据更新后，更新对应的资源\n            // this.data.addPropertyListener('details', (details) => {\n            //     updateHTML(details.label);\n            // });\n\n            // 生成针脚\n            const $content = this.shadowRoot.querySelector('.content')!;\n            $content.innerHTML = '';\n            if (blockDesc.createDynamicOutputPins) {\n                const outputList = blockDesc.createDynamicOutputPins(blockDesc, details);\n                outputList.forEach((pin, index) => {\n                    return $content?.appendChild(generateOutputPinHTML(pin, details.outputPins[index]));\n                });\n            } else {\n                blockDesc.outputPins.forEach((pin, index) => {\n                    return $content?.appendChild(generateOutputPinHTML(pin, details.outputPins[index]));\n                });\n            }\n            const $graph = (this.getRootNode() as any).host as GraphElement;\n            const uuid = this.getAttribute('node-uuid')!;\n            if (blockDesc.createDynamicInputPins) {\n                const inputList = blockDesc.createDynamicInputPins(blockDesc, details);\n                inputList.forEach((pin, index) => {\n                    return $content?.appendChild(generateInputPinHTML(pin, details.inputPins, index, uuid, $graph.getProperty('lines')));\n                });\n            } else {\n                blockDesc.inputPins.forEach((pin, index) => {\n                    return $content?.appendChild(generateInputPinHTML(pin, details.inputPins, index, uuid, $graph.getProperty('lines')));\n                });\n            }\n\n            // 绑定参数连接点的事件\n            const $paramList = this.shadowRoot.querySelectorAll(`v-graph-node-param`);\n            Array.prototype.forEach.call($paramList, ($param) => {\n                $param.addEventListener('mousedown', (event: MouseEvent) => {\n                    event.stopPropagation();\n                    event.preventDefault();\n\n                    const name = $param.getAttribute('name');\n                    if (!name) {\n                        return;\n                    }\n                    const paramDirection = $param.getAttribute('direction');\n                    if (paramDirection !== 'input' && paramDirection !== 'output') {\n                        return;\n                    }\n                    this.startConnect('curve', name, paramDirection);\n                });\n            });\n        },\n    };\n}\n\nconst STYLE = {\n    host(block: IBlockDescription) {\n        const config = block.style || {};\n        return /*css*/`\n:host *[hidden] {\n    display: none;\n}\n:host {\n    --font-color: ${config.fontColor || '#ccc'};\n    --font-color-hover: ${config.fontHoverColor || config.fontColor || '#ccc'};\n    --border-color: ${config.borderColor || 'white'};\n    --border-color-hover: ${config.borderHoverColor || config.borderColor || 'white'};\n    --shadow-color: ${config.shadowColor || '#ccc'};\n    --shadow-color-hover: ${config.shadowHoverColor || config.shadowColor || '#ccc'};\n    --background-color: ${config.backgroundColor || '#2b2b2bcc'};\n    --border-radius: 2px;\n\n    --header-height: 24px;\n    --header-background: ${config.headerColor || '#2b2b2bcc'};\n\n    --pin-height: 24px;\n\n    width: 200px;\n\n    color: var(--font-color);\n    cursor: default;\n\n}\n:host > section.wrapper {\n    margin: 10px;\n}\n:host(:hover) > section.wrapper, :host([selected]) > section.wrapper {\n    border-color: var(--border-color-hover);\n    color: var(--font-color-hover);\n    box-shadow: 0px 0px 7px 2px var(--shadow-color-hover);\n}\nsection.wrapper {\n    position: relative;\n    border-radius: var(--border-radius); \n    background: var(--background-color);\n    box-shadow: 0px 0px 7px 2px none;\n}\n:host(:hover) > section.wrapper::before {\n    content: '';\n    position: absolute;\n    top: 0;\n    left: 0;\n    right: 0;\n    bottom: 0;\n    z-index: 1;\n    border-radius: var(--border-radius); \n    box-shadow: 0px 0px 0px 1px var(--shadow-color-hover) inset;\n    pointer-events: none;\n}\n`;\n    },\n    header(block: IBlockDescription) {\n        const config = block.style || {};\n        return /*css*/`\nheader.title {\n    line-height: var(--header-height);\n\n    text-align: center;\n    border-radius: var(--border-radius);\n\n    ${config.secondaryColor ? `background: ${config.secondaryColor}; padding-left: 6px;` : ''}\n}\nheader.title > div {\n    padding: 0 10px;\n    height: 24px;\n    border-radius: var(--border-radius);\n    display: flex;\n    background: var(--header-background);\n}\n:host([expand]) header.title > div {\n    border-radius: var(--border-radius) var(--border-radius) 0 0;\n}\nheader.title > div > ui-label {\n    display: block;\n    padding: 0 10px;\n}\nheader.title > div > ui-icon {\n    display: block;\n}\nheader.title > div > svg {\n    fill: white;\n    display: block;\n    width: 10px;\n    transform: rotate(90deg);\n}\nheader.title > .quick-connect {\n    display: block;\n    padding: 0;\n    width: 12px;\n    height: 12px;\n    border-radius: 6px;\n    background: white;\n    position: absolute;\n    right: -6px;\n    top: 50%;\n    margin-top: -6px;\n    opacity: 0;\n    transition: opacity 0.3s;\n}\n:host(:hover) header.title > .quick-connect {\n    display: block;\n    opacity: 1;\n}\n        `;\n    },\n    pin: generatePinStyle,\n};\n\nexport function hasDeclareBlock(type: string) {\n    return blockMap.has(type);\n}\n\nexport function getDeclareBlock(type: string) {\n    return blockMap.get(type);\n}\n\nexport function removeDeclareBlock(type: string) {\n    const graph = '*';\n    const unknownOption = queryNode(graph, 'unknown');\n    registerNode(graph, type, unknownOption);\n    blockMap.delete(type);\n}\n\n// @ts-ignore\nwindow.removeDeclareBlock = removeDeclareBlock;\n\n/**\n * 注册一个 block 类型\n * @param block\n * @returns\n */\nexport function declareBlock(block: IBlockDescription) {\n    const graph = '*';\n    // if (blockMap.has(block.type)) {\n    //     console.warn(`Cannot declare duplicate block types: ${block.type}`);\n    //     return;\n    // }\n\n    // 合并 extend 数据\n    if (block.extend) {\n        const extend = blockMap.get(block.extend);\n        if (!extend) {\n            console.warn(`Inheritance data not found: ${block.extend}`);\n        } else {\n            completeBlockTarget(block, extend);\n        }\n    }\n\n    const options = generateBlockOption(block);\n    registerNode(graph, block.type, options);\n    blockMap.set(block.type, block);\n}\n\nexport function unDeclareBlock(type: string) {\n    const graph = '*';\n    // TODO 需要删除底层 block 节点\n    // unregisterNode(graph, type);\n    blockMap.delete(type);\n}\n\nexport function replaceDeclareBlock(searchType: string, block: IBlockDescription) {\n    if (blockMap.has(searchType)) {\n        unDeclareBlock(searchType);\n    }\n    declareBlock(block);\n}\n"]}