diff --git a/cypress/e2e/flowControlNodes.cy.js b/cypress/e2e/flowControlNodes.cy.js index 83abe4f..a4e92e3 100644 --- a/cypress/e2e/flowControlNodes.cy.js +++ b/cypress/e2e/flowControlNodes.cy.js @@ -154,6 +154,48 @@ const tests = [ delegate: false, development: false, }, + { + name: "Multi Gate", + value: String.raw` + Begin Object Class=/Script/BlueprintGraph.K2Node_MultiGate Name="K2Node_MultiGate_2" + NodePosX=-96 + NodePosY=-160 + NodeGuid=8D5767632F6C462B928E7F9A47E84AF3 + CustomProperties Pin (PinId=61334592A1B647A7888EDF804247FF70,PinName="execute",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=39200141C5D0415B825C28E3EC01A3F1,PinName="Out 0",Direction="EGPD_Output",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=EF36C23B68A44578B518B963E636D33C,PinName="Out 1",Direction="EGPD_Output",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=92FADDC1B07C45AC8BEF2FE42E13A638,PinName="Reset",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=363311761BC8401B8B26AD4B2D255749,PinName="IsRandom",PinType.PinCategory="bool",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=6F24605AD128461BB5652884D40E61E3,PinName="Loop",PinType.PinCategory="bool",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=E547C9E4961A43AD944E6877C2FF44D6,PinName="StartIndex",PinType.PinCategory="int",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,DefaultValue="-1",AutogeneratedDefaultValue="-1",PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + End Object + `, + color: Configuration.nodeColors.gray, + icon: SVGIcon.sequence, + pins: 7, + delegate: false, + development: false, + }, + { + name: "Do Once", + value: String.raw` + Begin Object Class=/Script/BlueprintGraph.K2Node_MacroInstance Name="K2Node_MacroInstance_1" + MacroGraphReference=(MacroGraph=/Script/Engine.EdGraph'"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:DoOnce"',GraphBlueprint=/Script/Engine.Blueprint'"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros"',GraphGuid=1281F54248A2ECB5B8B2C5B24AE6FDF4) + NodePosX=-416 + NodePosY=-112 + NodeGuid=A1831A1B85EF4E568E766FE3A3BCC5CD + CustomProperties Pin (PinId=5C24D82D7B084DFB841D17E5DF1CD8CF,PinName="execute",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=27242FC0B565448396C6A2DCD6BEDBD1,PinName="Reset",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=A8C0EB54974741248EA0B7B97FAC44DE,PinName="Start Closed",PinType.PinCategory="bool",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,DefaultValue="true",PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + CustomProperties Pin (PinId=3DE1FC03BFCD4ACF9AC7B99B89CEF465,PinName="Completed",Direction="EGPD_Output",PinType.PinCategory="exec",PinType.PinSubCategory="",PinType.PinSubCategoryObject=None,PinType.PinSubCategoryMemberReference=(),PinType.PinValueType=(),PinType.ContainerType=None,PinType.bIsReference=False,PinType.bIsConst=False,PinType.bIsWeakPointer=False,PinType.bIsUObjectWrapper=False,PinType.bSerializeAsSinglePrecisionFloat=False,LinkedTo=(K2Node_MultiGate_2 92FADDC1B07C45AC8BEF2FE42E13A638,),PersistentGuid=00000000000000000000000000000000,bHidden=False,bNotConnectable=False,bDefaultValueIsReadOnly=False,bDefaultValueIsIgnored=False,bAdvancedView=False,bOrphanedPin=False,) + End Object + `, + color: Configuration.nodeColors.gray, + icon: SVGIcon.doOnce, + pins: 4, + delegate: false, + development: false, + }, ] /** @type {Blueprint} */ diff --git a/dist/ueblueprint.js b/dist/ueblueprint.js index c003667..de58d9d 100755 --- a/dist/ueblueprint.js +++ b/dist/ueblueprint.js @@ -1,29 +1,29 @@ -/** - * @license - * Copyright 2019 Google LLC - * SPDX-License-Identifier: BSD-3-Clause - */ -const t$2=window,e$3=t$2.ShadowRoot&&(void 0===t$2.ShadyCSS||t$2.ShadyCSS.nativeShadow)&&"adoptedStyleSheets"in Document.prototype&&"replace"in CSSStyleSheet.prototype,s$3=Symbol(),n$3=new WeakMap;class o$3{constructor(t,e,n){if(this._$cssResult$=!0,n!==s$3)throw Error("CSSResult is not constructable. Use `unsafeCSS` or `css` instead.");this.cssText=t,this.t=e;}get styleSheet(){let t=this.o;const s=this.t;if(e$3&&void 0===t){const e=void 0!==s&&1===s.length;e&&(t=n$3.get(s)),void 0===t&&((this.o=t=new CSSStyleSheet).replaceSync(this.cssText),e&&n$3.set(s,t));}return t}toString(){return this.cssText}}const r$2=t=>new o$3("string"==typeof t?t:t+"",void 0,s$3),i$3=(t,...e)=>{const n=1===t.length?t[0]:e.reduce(((e,s,n)=>e+(t=>{if(!0===t._$cssResult$)return t.cssText;if("number"==typeof t)return t;throw Error("Value passed to 'css' function must be a 'css' function result: "+t+". Use 'unsafeCSS' to pass non-literal values, but take care to ensure page security.")})(s)+t[n+1]),t[0]);return new o$3(n,t,s$3)},S$1=(s,n)=>{e$3?s.adoptedStyleSheets=n.map((t=>t instanceof CSSStyleSheet?t:t.styleSheet)):n.forEach((e=>{const n=document.createElement("style"),o=t$2.litNonce;void 0!==o&&n.setAttribute("nonce",o),n.textContent=e.cssText,s.appendChild(n);}));},c$1=e$3?t=>t:t=>t instanceof CSSStyleSheet?(t=>{let e="";for(const s of t.cssRules)e+=s.cssText;return r$2(e)})(t):t; - -/** - * @license - * Copyright 2017 Google LLC - * SPDX-License-Identifier: BSD-3-Clause - */var s$2;const e$2=window,r$1=e$2.trustedTypes,h$1=r$1?r$1.emptyScript:"",o$2=e$2.reactiveElementPolyfillSupport,n$2={toAttribute(t,i){switch(i){case Boolean:t=t?h$1:null;break;case Object:case Array:t=null==t?t:JSON.stringify(t);}return t},fromAttribute(t,i){let s=t;switch(i){case Boolean:s=null!==t;break;case Number:s=null===t?null:Number(t);break;case Object:case Array:try{s=JSON.parse(t);}catch(t){s=null;}}return s}},a$1=(t,i)=>i!==t&&(i==i||t==t),l$2={attribute:!0,type:String,converter:n$2,reflect:!1,hasChanged:a$1};class d$1 extends HTMLElement{constructor(){super(),this._$Ei=new Map,this.isUpdatePending=!1,this.hasUpdated=!1,this._$El=null,this.u();}static addInitializer(t){var i;this.finalize(),(null!==(i=this.h)&&void 0!==i?i:this.h=[]).push(t);}static get observedAttributes(){this.finalize();const t=[];return this.elementProperties.forEach(((i,s)=>{const e=this._$Ep(s,i);void 0!==e&&(this._$Ev.set(e,s),t.push(e));})),t}static createProperty(t,i=l$2){if(i.state&&(i.attribute=!1),this.finalize(),this.elementProperties.set(t,i),!i.noAccessor&&!this.prototype.hasOwnProperty(t)){const s="symbol"==typeof t?Symbol():"__"+t,e=this.getPropertyDescriptor(t,s,i);void 0!==e&&Object.defineProperty(this.prototype,t,e);}}static getPropertyDescriptor(t,i,s){return {get(){return this[i]},set(e){const r=this[t];this[i]=e,this.requestUpdate(t,r,s);},configurable:!0,enumerable:!0}}static getPropertyOptions(t){return this.elementProperties.get(t)||l$2}static finalize(){if(this.hasOwnProperty("finalized"))return !1;this.finalized=!0;const t=Object.getPrototypeOf(this);if(t.finalize(),void 0!==t.h&&(this.h=[...t.h]),this.elementProperties=new Map(t.elementProperties),this._$Ev=new Map,this.hasOwnProperty("properties")){const t=this.properties,i=[...Object.getOwnPropertyNames(t),...Object.getOwnPropertySymbols(t)];for(const s of i)this.createProperty(s,t[s]);}return this.elementStyles=this.finalizeStyles(this.styles),!0}static finalizeStyles(i){const s=[];if(Array.isArray(i)){const e=new Set(i.flat(1/0).reverse());for(const i of e)s.unshift(c$1(i));}else void 0!==i&&s.push(c$1(i));return s}static _$Ep(t,i){const s=i.attribute;return !1===s?void 0:"string"==typeof s?s:"string"==typeof t?t.toLowerCase():void 0}u(){var t;this._$E_=new Promise((t=>this.enableUpdating=t)),this._$AL=new Map,this._$Eg(),this.requestUpdate(),null===(t=this.constructor.h)||void 0===t||t.forEach((t=>t(this)));}addController(t){var i,s;(null!==(i=this._$ES)&&void 0!==i?i:this._$ES=[]).push(t),void 0!==this.renderRoot&&this.isConnected&&(null===(s=t.hostConnected)||void 0===s||s.call(t));}removeController(t){var i;null===(i=this._$ES)||void 0===i||i.splice(this._$ES.indexOf(t)>>>0,1);}_$Eg(){this.constructor.elementProperties.forEach(((t,i)=>{this.hasOwnProperty(i)&&(this._$Ei.set(i,this[i]),delete this[i]);}));}createRenderRoot(){var t;const s=null!==(t=this.shadowRoot)&&void 0!==t?t:this.attachShadow(this.constructor.shadowRootOptions);return S$1(s,this.constructor.elementStyles),s}connectedCallback(){var t;void 0===this.renderRoot&&(this.renderRoot=this.createRenderRoot()),this.enableUpdating(!0),null===(t=this._$ES)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostConnected)||void 0===i?void 0:i.call(t)}));}enableUpdating(t){}disconnectedCallback(){var t;null===(t=this._$ES)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostDisconnected)||void 0===i?void 0:i.call(t)}));}attributeChangedCallback(t,i,s){this._$AK(t,s);}_$EO(t,i,s=l$2){var e;const r=this.constructor._$Ep(t,s);if(void 0!==r&&!0===s.reflect){const h=(void 0!==(null===(e=s.converter)||void 0===e?void 0:e.toAttribute)?s.converter:n$2).toAttribute(i,s.type);this._$El=t,null==h?this.removeAttribute(r):this.setAttribute(r,h),this._$El=null;}}_$AK(t,i){var s;const e=this.constructor,r=e._$Ev.get(t);if(void 0!==r&&this._$El!==r){const t=e.getPropertyOptions(r),h="function"==typeof t.converter?{fromAttribute:t.converter}:void 0!==(null===(s=t.converter)||void 0===s?void 0:s.fromAttribute)?t.converter:n$2;this._$El=r,this[r]=h.fromAttribute(i,t.type),this._$El=null;}}requestUpdate(t,i,s){let e=!0;void 0!==t&&(((s=s||this.constructor.getPropertyOptions(t)).hasChanged||a$1)(this[t],i)?(this._$AL.has(t)||this._$AL.set(t,i),!0===s.reflect&&this._$El!==t&&(void 0===this._$EC&&(this._$EC=new Map),this._$EC.set(t,s))):e=!1),!this.isUpdatePending&&e&&(this._$E_=this._$Ej());}async _$Ej(){this.isUpdatePending=!0;try{await this._$E_;}catch(t){Promise.reject(t);}const t=this.scheduleUpdate();return null!=t&&await t,!this.isUpdatePending}scheduleUpdate(){return this.performUpdate()}performUpdate(){var t;if(!this.isUpdatePending)return;this.hasUpdated,this._$Ei&&(this._$Ei.forEach(((t,i)=>this[i]=t)),this._$Ei=void 0);let i=!1;const s=this._$AL;try{i=this.shouldUpdate(s),i?(this.willUpdate(s),null===(t=this._$ES)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostUpdate)||void 0===i?void 0:i.call(t)})),this.update(s)):this._$Ek();}catch(t){throw i=!1,this._$Ek(),t}i&&this._$AE(s);}willUpdate(t){}_$AE(t){var i;null===(i=this._$ES)||void 0===i||i.forEach((t=>{var i;return null===(i=t.hostUpdated)||void 0===i?void 0:i.call(t)})),this.hasUpdated||(this.hasUpdated=!0,this.firstUpdated(t)),this.updated(t);}_$Ek(){this._$AL=new Map,this.isUpdatePending=!1;}get updateComplete(){return this.getUpdateComplete()}getUpdateComplete(){return this._$E_}shouldUpdate(t){return !0}update(t){void 0!==this._$EC&&(this._$EC.forEach(((t,i)=>this._$EO(i,this[i],t))),this._$EC=void 0),this._$Ek();}updated(t){}firstUpdated(t){}}d$1.finalized=!0,d$1.elementProperties=new Map,d$1.elementStyles=[],d$1.shadowRootOptions={mode:"open"},null==o$2||o$2({ReactiveElement:d$1}),(null!==(s$2=e$2.reactiveElementVersions)&&void 0!==s$2?s$2:e$2.reactiveElementVersions=[]).push("1.4.2"); - -/** - * @license - * Copyright 2017 Google LLC - * SPDX-License-Identifier: BSD-3-Clause - */ -var t$1;const i$2=window,s$1=i$2.trustedTypes,e$1=s$1?s$1.createPolicy("lit-html",{createHTML:t=>t}):void 0,o$1=`lit$${(Math.random()+"").slice(9)}$`,n$1="?"+o$1,l$1=`<${n$1}>`,h=document,r=(t="")=>h.createComment(t),d=t=>null===t||"object"!=typeof t&&"function"!=typeof t,u=Array.isArray,c=t=>u(t)||"function"==typeof(null==t?void 0:t[Symbol.iterator]),v=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,a=/-->/g,f=/>/g,_=RegExp(">|[ \t\n\f\r](?:([^\\s\"'>=/]+)([ \t\n\f\r]*=[ \t\n\f\r]*(?:[^ \t\n\f\r\"'`<>=]|(\"|')|))|$)","g"),m=/'/g,p=/"/g,$=/^(?:script|style|textarea|title)$/i,g=t=>(i,...s)=>({_$litType$:t,strings:i,values:s}),y=g(1),x=Symbol.for("lit-noChange"),b=Symbol.for("lit-nothing"),T=new WeakMap,A=h.createTreeWalker(h,129,null,!1),E=(t,i)=>{const s=t.length-1,n=[];let h,r=2===i?"":"",d=v;for(let i=0;i"===u[0]?(d=null!=h?h:v,c=-1):void 0===u[1]?c=-2:(c=d.lastIndex-u[2].length,e=u[1],d=void 0===u[3]?_:'"'===u[3]?p:m):d===p||d===m?d=_:d===a||d===f?d=v:(d=_,h=void 0);const y=d===_&&t[i+1].startsWith("/>")?" ":"";r+=d===v?s+l$1:c>=0?(n.push(e),s.slice(0,c)+"$lit$"+s.slice(c)+o$1+y):s+o$1+(-2===c?(n.push(void 0),i):y);}const u=r+(t[s]||"")+(2===i?"":"");if(!Array.isArray(t)||!t.hasOwnProperty("raw"))throw Error("invalid template strings array");return [void 0!==e$1?e$1.createHTML(u):u,n]};class C{constructor({strings:t,_$litType$:i},e){let l;this.parts=[];let h=0,d=0;const u=t.length-1,c=this.parts,[v,a]=E(t,i);if(this.el=C.createElement(v,e),A.currentNode=this.el.content,2===i){const t=this.el.content,i=t.firstChild;i.remove(),t.append(...i.childNodes);}for(;null!==(l=A.nextNode())&&c.length0){l.textContent=s$1?s$1.emptyScript:"";for(let s=0;s2||""!==s[0]||""!==s[1]?(this._$AH=Array(s.length-1).fill(new String),this.strings=s):this._$AH=b;}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(t,i=this,s,e){const o=this.strings;let n=!1;if(void 0===o)t=P$1(this,t,i,0),n=!d(t)||t!==this._$AH&&t!==x,n&&(this._$AH=t);else {const e=t;let l,h;for(t=o[0],l=0;l{var e,o;const n=null!==(e=null==s?void 0:s.renderBefore)&&void 0!==e?e:i;let l=n._$litPart$;if(void 0===l){const t=null!==(o=null==s?void 0:s.renderBefore)&&void 0!==o?o:null;n._$litPart$=l=new N(i.insertBefore(r(),t),t,void 0,null!=s?s:{});}return l._$AI(t),l}; - -/** - * @license - * Copyright 2017 Google LLC - * SPDX-License-Identifier: BSD-3-Clause - */var l,o;class s extends d$1{constructor(){super(...arguments),this.renderOptions={host:this},this._$Do=void 0;}createRenderRoot(){var t,e;const i=super.createRenderRoot();return null!==(t=(e=this.renderOptions).renderBefore)&&void 0!==t||(e.renderBefore=i.firstChild),i}update(t){const i=this.render();this.hasUpdated||(this.renderOptions.isConnected=this.isConnected),super.update(t),this._$Do=Z(i,this.renderRoot,this.renderOptions);}connectedCallback(){var t;super.connectedCallback(),null===(t=this._$Do)||void 0===t||t.setConnected(!0);}disconnectedCallback(){var t;super.disconnectedCallback(),null===(t=this._$Do)||void 0===t||t.setConnected(!1);}render(){return x}}s.finalized=!0,s._$litElement$=!0,null===(l=globalThis.litElementHydrateSupport)||void 0===l||l.call(globalThis,{LitElement:s});const n=globalThis.litElementPolyfillSupport;null==n||n({LitElement:s});(null!==(o=globalThis.litElementVersions)&&void 0!==o?o:globalThis.litElementVersions=[]).push("3.2.2"); - +/** + * @license + * Copyright 2019 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + */ +const t$2=window,e$3=t$2.ShadowRoot&&(void 0===t$2.ShadyCSS||t$2.ShadyCSS.nativeShadow)&&"adoptedStyleSheets"in Document.prototype&&"replace"in CSSStyleSheet.prototype,s$3=Symbol(),n$3=new WeakMap;class o$3{constructor(t,e,n){if(this._$cssResult$=!0,n!==s$3)throw Error("CSSResult is not constructable. Use `unsafeCSS` or `css` instead.");this.cssText=t,this.t=e;}get styleSheet(){let t=this.o;const s=this.t;if(e$3&&void 0===t){const e=void 0!==s&&1===s.length;e&&(t=n$3.get(s)),void 0===t&&((this.o=t=new CSSStyleSheet).replaceSync(this.cssText),e&&n$3.set(s,t));}return t}toString(){return this.cssText}}const r$2=t=>new o$3("string"==typeof t?t:t+"",void 0,s$3),i$3=(t,...e)=>{const n=1===t.length?t[0]:e.reduce(((e,s,n)=>e+(t=>{if(!0===t._$cssResult$)return t.cssText;if("number"==typeof t)return t;throw Error("Value passed to 'css' function must be a 'css' function result: "+t+". Use 'unsafeCSS' to pass non-literal values, but take care to ensure page security.")})(s)+t[n+1]),t[0]);return new o$3(n,t,s$3)},S$1=(s,n)=>{e$3?s.adoptedStyleSheets=n.map((t=>t instanceof CSSStyleSheet?t:t.styleSheet)):n.forEach((e=>{const n=document.createElement("style"),o=t$2.litNonce;void 0!==o&&n.setAttribute("nonce",o),n.textContent=e.cssText,s.appendChild(n);}));},c$1=e$3?t=>t:t=>t instanceof CSSStyleSheet?(t=>{let e="";for(const s of t.cssRules)e+=s.cssText;return r$2(e)})(t):t; + +/** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + */var s$2;const e$2=window,r$1=e$2.trustedTypes,h$1=r$1?r$1.emptyScript:"",o$2=e$2.reactiveElementPolyfillSupport,n$2={toAttribute(t,i){switch(i){case Boolean:t=t?h$1:null;break;case Object:case Array:t=null==t?t:JSON.stringify(t);}return t},fromAttribute(t,i){let s=t;switch(i){case Boolean:s=null!==t;break;case Number:s=null===t?null:Number(t);break;case Object:case Array:try{s=JSON.parse(t);}catch(t){s=null;}}return s}},a$1=(t,i)=>i!==t&&(i==i||t==t),l$2={attribute:!0,type:String,converter:n$2,reflect:!1,hasChanged:a$1};class d$1 extends HTMLElement{constructor(){super(),this._$Ei=new Map,this.isUpdatePending=!1,this.hasUpdated=!1,this._$El=null,this.u();}static addInitializer(t){var i;this.finalize(),(null!==(i=this.h)&&void 0!==i?i:this.h=[]).push(t);}static get observedAttributes(){this.finalize();const t=[];return this.elementProperties.forEach(((i,s)=>{const e=this._$Ep(s,i);void 0!==e&&(this._$Ev.set(e,s),t.push(e));})),t}static createProperty(t,i=l$2){if(i.state&&(i.attribute=!1),this.finalize(),this.elementProperties.set(t,i),!i.noAccessor&&!this.prototype.hasOwnProperty(t)){const s="symbol"==typeof t?Symbol():"__"+t,e=this.getPropertyDescriptor(t,s,i);void 0!==e&&Object.defineProperty(this.prototype,t,e);}}static getPropertyDescriptor(t,i,s){return {get(){return this[i]},set(e){const r=this[t];this[i]=e,this.requestUpdate(t,r,s);},configurable:!0,enumerable:!0}}static getPropertyOptions(t){return this.elementProperties.get(t)||l$2}static finalize(){if(this.hasOwnProperty("finalized"))return !1;this.finalized=!0;const t=Object.getPrototypeOf(this);if(t.finalize(),void 0!==t.h&&(this.h=[...t.h]),this.elementProperties=new Map(t.elementProperties),this._$Ev=new Map,this.hasOwnProperty("properties")){const t=this.properties,i=[...Object.getOwnPropertyNames(t),...Object.getOwnPropertySymbols(t)];for(const s of i)this.createProperty(s,t[s]);}return this.elementStyles=this.finalizeStyles(this.styles),!0}static finalizeStyles(i){const s=[];if(Array.isArray(i)){const e=new Set(i.flat(1/0).reverse());for(const i of e)s.unshift(c$1(i));}else void 0!==i&&s.push(c$1(i));return s}static _$Ep(t,i){const s=i.attribute;return !1===s?void 0:"string"==typeof s?s:"string"==typeof t?t.toLowerCase():void 0}u(){var t;this._$E_=new Promise((t=>this.enableUpdating=t)),this._$AL=new Map,this._$Eg(),this.requestUpdate(),null===(t=this.constructor.h)||void 0===t||t.forEach((t=>t(this)));}addController(t){var i,s;(null!==(i=this._$ES)&&void 0!==i?i:this._$ES=[]).push(t),void 0!==this.renderRoot&&this.isConnected&&(null===(s=t.hostConnected)||void 0===s||s.call(t));}removeController(t){var i;null===(i=this._$ES)||void 0===i||i.splice(this._$ES.indexOf(t)>>>0,1);}_$Eg(){this.constructor.elementProperties.forEach(((t,i)=>{this.hasOwnProperty(i)&&(this._$Ei.set(i,this[i]),delete this[i]);}));}createRenderRoot(){var t;const s=null!==(t=this.shadowRoot)&&void 0!==t?t:this.attachShadow(this.constructor.shadowRootOptions);return S$1(s,this.constructor.elementStyles),s}connectedCallback(){var t;void 0===this.renderRoot&&(this.renderRoot=this.createRenderRoot()),this.enableUpdating(!0),null===(t=this._$ES)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostConnected)||void 0===i?void 0:i.call(t)}));}enableUpdating(t){}disconnectedCallback(){var t;null===(t=this._$ES)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostDisconnected)||void 0===i?void 0:i.call(t)}));}attributeChangedCallback(t,i,s){this._$AK(t,s);}_$EO(t,i,s=l$2){var e;const r=this.constructor._$Ep(t,s);if(void 0!==r&&!0===s.reflect){const h=(void 0!==(null===(e=s.converter)||void 0===e?void 0:e.toAttribute)?s.converter:n$2).toAttribute(i,s.type);this._$El=t,null==h?this.removeAttribute(r):this.setAttribute(r,h),this._$El=null;}}_$AK(t,i){var s;const e=this.constructor,r=e._$Ev.get(t);if(void 0!==r&&this._$El!==r){const t=e.getPropertyOptions(r),h="function"==typeof t.converter?{fromAttribute:t.converter}:void 0!==(null===(s=t.converter)||void 0===s?void 0:s.fromAttribute)?t.converter:n$2;this._$El=r,this[r]=h.fromAttribute(i,t.type),this._$El=null;}}requestUpdate(t,i,s){let e=!0;void 0!==t&&(((s=s||this.constructor.getPropertyOptions(t)).hasChanged||a$1)(this[t],i)?(this._$AL.has(t)||this._$AL.set(t,i),!0===s.reflect&&this._$El!==t&&(void 0===this._$EC&&(this._$EC=new Map),this._$EC.set(t,s))):e=!1),!this.isUpdatePending&&e&&(this._$E_=this._$Ej());}async _$Ej(){this.isUpdatePending=!0;try{await this._$E_;}catch(t){Promise.reject(t);}const t=this.scheduleUpdate();return null!=t&&await t,!this.isUpdatePending}scheduleUpdate(){return this.performUpdate()}performUpdate(){var t;if(!this.isUpdatePending)return;this.hasUpdated,this._$Ei&&(this._$Ei.forEach(((t,i)=>this[i]=t)),this._$Ei=void 0);let i=!1;const s=this._$AL;try{i=this.shouldUpdate(s),i?(this.willUpdate(s),null===(t=this._$ES)||void 0===t||t.forEach((t=>{var i;return null===(i=t.hostUpdate)||void 0===i?void 0:i.call(t)})),this.update(s)):this._$Ek();}catch(t){throw i=!1,this._$Ek(),t}i&&this._$AE(s);}willUpdate(t){}_$AE(t){var i;null===(i=this._$ES)||void 0===i||i.forEach((t=>{var i;return null===(i=t.hostUpdated)||void 0===i?void 0:i.call(t)})),this.hasUpdated||(this.hasUpdated=!0,this.firstUpdated(t)),this.updated(t);}_$Ek(){this._$AL=new Map,this.isUpdatePending=!1;}get updateComplete(){return this.getUpdateComplete()}getUpdateComplete(){return this._$E_}shouldUpdate(t){return !0}update(t){void 0!==this._$EC&&(this._$EC.forEach(((t,i)=>this._$EO(i,this[i],t))),this._$EC=void 0),this._$Ek();}updated(t){}firstUpdated(t){}}d$1.finalized=!0,d$1.elementProperties=new Map,d$1.elementStyles=[],d$1.shadowRootOptions={mode:"open"},null==o$2||o$2({ReactiveElement:d$1}),(null!==(s$2=e$2.reactiveElementVersions)&&void 0!==s$2?s$2:e$2.reactiveElementVersions=[]).push("1.4.2"); + +/** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + */ +var t$1;const i$2=window,s$1=i$2.trustedTypes,e$1=s$1?s$1.createPolicy("lit-html",{createHTML:t=>t}):void 0,o$1=`lit$${(Math.random()+"").slice(9)}$`,n$1="?"+o$1,l$1=`<${n$1}>`,h=document,r=(t="")=>h.createComment(t),d=t=>null===t||"object"!=typeof t&&"function"!=typeof t,u=Array.isArray,c=t=>u(t)||"function"==typeof(null==t?void 0:t[Symbol.iterator]),v=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,a=/-->/g,f=/>/g,_=RegExp(">|[ \t\n\f\r](?:([^\\s\"'>=/]+)([ \t\n\f\r]*=[ \t\n\f\r]*(?:[^ \t\n\f\r\"'`<>=]|(\"|')|))|$)","g"),m=/'/g,p=/"/g,$=/^(?:script|style|textarea|title)$/i,g=t=>(i,...s)=>({_$litType$:t,strings:i,values:s}),y=g(1),x=Symbol.for("lit-noChange"),b=Symbol.for("lit-nothing"),T=new WeakMap,A=h.createTreeWalker(h,129,null,!1),E=(t,i)=>{const s=t.length-1,n=[];let h,r=2===i?"":"",d=v;for(let i=0;i"===u[0]?(d=null!=h?h:v,c=-1):void 0===u[1]?c=-2:(c=d.lastIndex-u[2].length,e=u[1],d=void 0===u[3]?_:'"'===u[3]?p:m):d===p||d===m?d=_:d===a||d===f?d=v:(d=_,h=void 0);const y=d===_&&t[i+1].startsWith("/>")?" ":"";r+=d===v?s+l$1:c>=0?(n.push(e),s.slice(0,c)+"$lit$"+s.slice(c)+o$1+y):s+o$1+(-2===c?(n.push(void 0),i):y);}const u=r+(t[s]||"")+(2===i?"":"");if(!Array.isArray(t)||!t.hasOwnProperty("raw"))throw Error("invalid template strings array");return [void 0!==e$1?e$1.createHTML(u):u,n]};class C{constructor({strings:t,_$litType$:i},e){let l;this.parts=[];let h=0,d=0;const u=t.length-1,c=this.parts,[v,a]=E(t,i);if(this.el=C.createElement(v,e),A.currentNode=this.el.content,2===i){const t=this.el.content,i=t.firstChild;i.remove(),t.append(...i.childNodes);}for(;null!==(l=A.nextNode())&&c.length0){l.textContent=s$1?s$1.emptyScript:"";for(let s=0;s2||""!==s[0]||""!==s[1]?(this._$AH=Array(s.length-1).fill(new String),this.strings=s):this._$AH=b;}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(t,i=this,s,e){const o=this.strings;let n=!1;if(void 0===o)t=P$1(this,t,i,0),n=!d(t)||t!==this._$AH&&t!==x,n&&(this._$AH=t);else {const e=t;let l,h;for(t=o[0],l=0;l{var e,o;const n=null!==(e=null==s?void 0:s.renderBefore)&&void 0!==e?e:i;let l=n._$litPart$;if(void 0===l){const t=null!==(o=null==s?void 0:s.renderBefore)&&void 0!==o?o:null;n._$litPart$=l=new N(i.insertBefore(r(),t),t,void 0,null!=s?s:{});}return l._$AI(t),l}; + +/** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + */var l,o;class s extends d$1{constructor(){super(...arguments),this.renderOptions={host:this},this._$Do=void 0;}createRenderRoot(){var t,e;const i=super.createRenderRoot();return null!==(t=(e=this.renderOptions).renderBefore)&&void 0!==t||(e.renderBefore=i.firstChild),i}update(t){const i=this.render();this.hasUpdated||(this.renderOptions.isConnected=this.isConnected),super.update(t),this._$Do=Z(i,this.renderRoot,this.renderOptions);}connectedCallback(){var t;super.connectedCallback(),null===(t=this._$Do)||void 0===t||t.setConnected(!0);}disconnectedCallback(){var t;super.disconnectedCallback(),null===(t=this._$Do)||void 0===t||t.setConnected(!1);}render(){return x}}s.finalized=!0,s._$litElement$=!0,null===(l=globalThis.litElementHydrateSupport)||void 0===l||l.call(globalThis,{LitElement:s});const n=globalThis.litElementPolyfillSupport;null==n||n({LitElement:s});(null!==(o=globalThis.litElementVersions)&&void 0!==o?o:globalThis.litElementVersions=[]).push("3.2.2"); + /** * @typedef {import("./element/NodeElement").default} NodeElement * @typedef {import("./element/PinElement").default} PinElement @@ -100,6 +100,7 @@ class Configuration { componentBoundEvent: "/Script/BlueprintGraph.K2Node_ComponentBoundEvent", customEvent: "/Script/BlueprintGraph.K2Node_CustomEvent", doN: "/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:Do N", + doOnce: "/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:DoOnce", dynamicCast: "/Script/BlueprintGraph.K2Node_DynamicCast", enum: "/Script/CoreUObject.Enum", enumLiteral: "/Script/BlueprintGraph.K2Node_EnumLiteral", @@ -122,6 +123,7 @@ class Configuration { makeArray: "/Script/BlueprintGraph.K2Node_MakeArray", makeMap: "/Script/BlueprintGraph.K2Node_MakeMap", makeSet: "/Script/BlueprintGraph.K2Node_MakeSet", + multiGate: "/Script/BlueprintGraph.K2Node_MultiGate", pawn: "/Script/Engine.Pawn", promotableOperator: "/Script/BlueprintGraph.K2Node_PromotableOperator", reverseForEachLoop: "/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ReverseForEachLoop", @@ -285,76 +287,76 @@ class Configuration { "NumLock": "NumLock", "ScrollLock": "ScrollLock", } -} - -/** @typedef {import("../Blueprint").default} Blueprint */ - -/** @template {HTMLElement} T */ -class IInput { - - /** @type {T} */ - #target - get target() { - return this.#target - } - - /** @type {Blueprint} */ - #blueprint - get blueprint() { - return this.#blueprint - } - - /** @type {Object} */ - options - - - listenHandler = () => this.listenEvents() - unlistenHandler = () => this.unlistenEvents() - - /** - * @param {T} target - * @param {Blueprint} blueprint - * @param {Object} options - */ - constructor(target, blueprint, options = {}) { - options.consumeEvent ??= false; - options.listenOnFocus ??= false; - options.unlistenOnTextEdit ??= false; - this.#target = target; - this.#blueprint = blueprint; - this.options = options; - } - - setup() { - if (this.options.listenOnFocus) { - this.blueprint.addEventListener(Configuration.focusEventName.begin, this.listenHandler); - this.blueprint.addEventListener(Configuration.focusEventName.end, this.unlistenHandler); - } - if (this.options.unlistenOnTextEdit) { - this.blueprint.addEventListener(Configuration.editTextEventName.begin, this.unlistenHandler); - this.blueprint.addEventListener(Configuration.editTextEventName.end, this.listenHandler); - } - if (this.blueprint.focused) { - this.listenEvents(); - } - } - - cleanup() { - this.unlistenEvents(); - this.blueprint.removeEventListener(Configuration.focusEventName.begin, this.listenHandler); - this.blueprint.removeEventListener(Configuration.focusEventName.end, this.unlistenHandler); - this.blueprint.removeEventListener(Configuration.editTextEventName.begin, this.unlistenHandler); - this.blueprint.removeEventListener(Configuration.editTextEventName.end, this.listenHandler); - } - - /* Subclasses will probabily override the following methods */ - listenEvents() { - } - - unlistenEvents() { - } -} - +} + +/** @typedef {import("../Blueprint").default} Blueprint */ + +/** @template {HTMLElement} T */ +class IInput { + + /** @type {T} */ + #target + get target() { + return this.#target + } + + /** @type {Blueprint} */ + #blueprint + get blueprint() { + return this.#blueprint + } + + /** @type {Object} */ + options + + + listenHandler = () => this.listenEvents() + unlistenHandler = () => this.unlistenEvents() + + /** + * @param {T} target + * @param {Blueprint} blueprint + * @param {Object} options + */ + constructor(target, blueprint, options = {}) { + options.consumeEvent ??= false; + options.listenOnFocus ??= false; + options.unlistenOnTextEdit ??= false; + this.#target = target; + this.#blueprint = blueprint; + this.options = options; + } + + setup() { + if (this.options.listenOnFocus) { + this.blueprint.addEventListener(Configuration.focusEventName.begin, this.listenHandler); + this.blueprint.addEventListener(Configuration.focusEventName.end, this.unlistenHandler); + } + if (this.options.unlistenOnTextEdit) { + this.blueprint.addEventListener(Configuration.editTextEventName.begin, this.unlistenHandler); + this.blueprint.addEventListener(Configuration.editTextEventName.end, this.listenHandler); + } + if (this.blueprint.focused) { + this.listenEvents(); + } + } + + cleanup() { + this.unlistenEvents(); + this.blueprint.removeEventListener(Configuration.focusEventName.begin, this.listenHandler); + this.blueprint.removeEventListener(Configuration.focusEventName.end, this.unlistenHandler); + this.blueprint.removeEventListener(Configuration.editTextEventName.begin, this.unlistenHandler); + this.blueprint.removeEventListener(Configuration.editTextEventName.end, this.listenHandler); + } + + /* Subclasses will probabily override the following methods */ + listenEvents() { + } + + unlistenEvents() { + } +} + /** * @typedef {import("../entity/IEntity").default} IEntity * @typedef {import("../entity/IEntity").AnyValue} AnyValue @@ -392,37 +394,37 @@ class SerializerFactory { // @ts-expect-error return SerializerFactory.#serializers.get(entity) } -} - -/** @typedef {import("./IEntity").AttributeDeclarations} AttributeDeclarations */ - -class SubAttributesDeclaration { - - /** @param {AttributeDeclarations} attributes */ - constructor(attributes) { - this.attributes = attributes; - } -} - -/** @typedef {import("./IEntity").AnyValueConstructor<*>} AnyValueConstructor */ - -class UnionType { - - #types - get types() { - return this.#types - } - - /** @param {...AnyValueConstructor} types */ - constructor(...types) { - this.#types = types; - } - - getFirstType() { - return this.#types[0] - } -} - +} + +/** @typedef {import("./IEntity").AttributeDeclarations} AttributeDeclarations */ + +class SubAttributesDeclaration { + + /** @param {AttributeDeclarations} attributes */ + constructor(attributes) { + this.attributes = attributes; + } +} + +/** @typedef {import("./IEntity").AnyValueConstructor<*>} AnyValueConstructor */ + +class UnionType { + + #types + get types() { + return this.#types + } + + /** @param {...AnyValueConstructor} types */ + constructor(...types) { + this.#types = types; + } + + getFirstType() { + return this.#types[0] + } +} + /** * @typedef {import("./element/IElement").default} IElement * @typedef {import("./entity/IEntity").AnyValue} AnyValue @@ -748,7 +750,7 @@ class Utility { } /** @param {String} value */ - static formatStringName(value) { + static formatStringName(value = "") { return value // Remove leading b (for boolean values) or newlines .replace(/^\s*b/, "") @@ -846,258 +848,258 @@ class Utility { }; requestAnimationFrame(doAnimation); } -} - -/** - * @typedef {(entity: IEntity) => AnyValue} ValueSupplier - * @typedef {(entity: IEntity) => AnyValueConstructor} TypeSupplier - * @typedef {IEntity | String | Number | BigInt | Boolean} AnySimpleValue - * @typedef {AnySimpleValue | AnySimpleValue[]} AnyValue - * @typedef {{ - * [key: String]: AttributeInformation | AnyValue | SubAttributesDeclaration - * }} AttributeDeclarations - * @typedef {typeof IEntity} EntityConstructor - * @typedef {{ - * type?: AnyValueConstructor | AnyValueConstructor[] | UnionType | TypeSupplier, - * value?: AnyValue | ValueSupplier, - * showDefault?: Boolean, - * nullable?: Boolean, - * ignored?: Boolean, - * serialized?: Boolean, - * expected?: Boolean, - * predicate?: (value: AnyValue) => Boolean, - * }} AttributeInformation - */ - -/** - * @template {AnyValue} T - * @typedef {(new () => T) | EntityConstructor | StringConstructor | NumberConstructor | BigIntConstructor - * | BooleanConstructor | ArrayConstructor} AnyValueConstructor - */ - -class IEntity { - - /** @type {AttributeDeclarations} */ - static attributes = {} - static defaultAttribute = { - showDefault: true, - nullable: false, - ignored: false, - serialized: false, - expected: false, - } - - constructor(values = {}, suppressWarns = false) { - /** - * @param {Object} target - * @param {Object} attributes - * @param {Object} values - * @param {String} prefix - */ - const defineAllAttributes = (target, attributes, values = {}, prefix = "") => { - const valuesNames = Object.keys(values); - const attributesNames = Object.keys(attributes); - const allAttributesNames = Utility.mergeArrays(attributesNames, valuesNames); - for (let attributeName of allAttributesNames) { - let value = Utility.objectGet(values, [attributeName]); - /** @type {AttributeInformation} */ - let attribute = attributes[attributeName]; - - if (attribute instanceof SubAttributesDeclaration) { - target[attributeName] = {}; - defineAllAttributes( - target[attributeName], - attribute.attributes, - values[attributeName], - attributeName + "." - ); - continue - } - - if (!suppressWarns) { - if (!(attributeName in attributes)) { - console.warn( - `UEBlueprint: Attribute ${prefix}${attributeName} in the serialized data is not defined in ` - + `${this.constructor.name}.attributes` - ); - } else if ( - valuesNames.length > 0 - && !(attributeName in values) - && !(!attribute.showDefault || attribute.ignored) - ) { - console.warn( - `UEBlueprint: ${this.constructor.name} will add attribute ${prefix}${attributeName} not ` - + "defined in the serialized data" - ); - } - } - - if (!attribute) { - // Remember attributeName can come from the values and be not defined in the attributes - target[attributeName] = value; - continue - } - - let defaultValue = attribute.value; - let defaultType = attribute.type; - if (attribute.serialized && defaultType instanceof Function) { - // If the attribute is serialized, the type must contain a function providing the type - defaultType = /** @type {TypeSupplier} */(defaultType)(this); - } - if (defaultType instanceof Array) { - defaultType = Array; - } - if (defaultValue instanceof Function) { - defaultValue = defaultValue(this); - } - if (defaultType === undefined) { - defaultType = Utility.getType(defaultValue); - } - const assignAttribute = !attribute.predicate - ? v => target[attributeName] = v - : v => { - Object.defineProperties(target, { - ["#" + attributeName]: { - writable: true, - enumerable: false, - }, - [attributeName]: { - enumerable: true, - get() { - return this["#" + attributeName] - }, - set(v) { - if (!attribute.predicate?.(v)) { - console.warn( - `UEBlueprint: Tried to assign attribute ${prefix}${attributeName} to ` - + `${this.constructor.name} not satisfying the predicate` - ); - return - } - this["#" + attributeName] = v; - } - }, - }); - this[attributeName] = v; - }; - - if (value !== undefined) { - // Remember value can still be null - if (value?.constructor === String && attribute.serialized && defaultType !== String) { - value = SerializerFactory - .getSerializer(/** @type {AnyValueConstructor<*>} */(defaultType)) - .deserialize(/** @type {String} */(value)); - } - assignAttribute(Utility.sanitize(value, /** @type {AnyValueConstructor<*>} */(defaultType))); - continue // We have a value, need nothing more - } - if (defaultType instanceof UnionType) { - if (defaultValue != undefined) { - defaultType = defaultType.types.find( - type => defaultValue instanceof type || defaultValue.constructor == type - ) ?? defaultType.getFirstType(); - } else { - defaultType = defaultType.getFirstType(); - } - } - if (defaultValue === undefined) { - defaultValue = Utility.sanitize(new /** @type {AnyValueConstructor<*>} */(defaultType)()); - } - if (!attribute.showDefault) { - assignAttribute(undefined); // Declare undefined to preserve the order of attributes - continue - } - if (attribute.serialized) { - if (defaultType !== String && defaultValue.constructor === String) { - defaultValue = SerializerFactory - .getSerializer(/** @type {AnyValueConstructor<*>} */(defaultType)) - .deserialize(defaultValue); - } - } - assignAttribute(Utility.sanitize( - /** @type {AnyValue} */(defaultValue), - /** @type {AnyValueConstructor} */(defaultType) - )); - } - }; - const attributes = /** @type {typeof IEntity} */(this.constructor).attributes; - if (values.constructor !== Object && Object.keys(attributes).length === 1) { - // Where there is just one attribute, option can be the value of that attribute - values = { - [Object.keys(attributes)[0]]: values - }; - } - defineAllAttributes(this, attributes, values); - } - - /** @param {AttributeDeclarations} attributes */ - static cleanupAttributes(attributes, prefix = "") { - for (const attributeName in attributes) { - if (attributes[attributeName] instanceof SubAttributesDeclaration) { - this.cleanupAttributes( - /** @type {SubAttributesDeclaration} */(attributes[attributeName]).attributes, - prefix + "." + attributeName - ); - continue - } - if (attributes[attributeName].constructor !== Object) { - attributes[attributeName] = { - value: attributes[attributeName], - }; - } - const attribute = /** @type {AttributeInformation} */(attributes[attributeName]); - if (attribute.type === undefined && !(attribute.value instanceof Function)) { - attribute.type = Utility.getType(attribute.value); - } - attributes[attributeName] = { - ...IEntity.defaultAttribute, - ...attribute, - }; - if (attribute.value === undefined && attribute.type === undefined) { - throw new Error( - `UEBlueprint: Expected either "type" or "value" property in ${this.name} attribute ${prefix}` - + attributeName - ) - } - if (attribute.value === null) { - attributes[attributeName].nullable = true; - } - } - } - - static isValueOfType(value, type) { - return value != null && (value instanceof type || value.constructor === type) - } - - static expectsAllKeys() { - return !Object.values(this.attributes) - .filter(/** @param {AttributeInformation} attribute */attribute => !attribute.ignored) - .some(/** @param {AttributeInformation} attribute */attribute => !attribute.expected) - } - - unexpectedKeys() { - return Object.keys(this).length - - Object.keys(/** @type {typeof IEntity} */(this.constructor).attributes).length - } - - /** @param {IEntity} other */ - equals(other) { - const thisKeys = Object.keys(this); - const otherKeys = Object.keys(this); - if (thisKeys.length != otherKeys.length) { - return false - } - for (const key of thisKeys) { - if (this[key] instanceof IEntity && !this[key].equals(other[key])) { - return false - } else if (!Utility.equals(this[key], other[key])) { - return false - } - } - return true - } -} - +} + +/** + * @typedef {(entity: IEntity) => AnyValue} ValueSupplier + * @typedef {(entity: IEntity) => AnyValueConstructor} TypeSupplier + * @typedef {IEntity | String | Number | BigInt | Boolean} AnySimpleValue + * @typedef {AnySimpleValue | AnySimpleValue[]} AnyValue + * @typedef {{ + * [key: String]: AttributeInformation | AnyValue | SubAttributesDeclaration + * }} AttributeDeclarations + * @typedef {typeof IEntity} EntityConstructor + * @typedef {{ + * type?: AnyValueConstructor | AnyValueConstructor[] | UnionType | TypeSupplier, + * value?: AnyValue | ValueSupplier, + * showDefault?: Boolean, + * nullable?: Boolean, + * ignored?: Boolean, + * serialized?: Boolean, + * expected?: Boolean, + * predicate?: (value: AnyValue) => Boolean, + * }} AttributeInformation + */ + +/** + * @template {AnyValue} T + * @typedef {(new () => T) | EntityConstructor | StringConstructor | NumberConstructor | BigIntConstructor + * | BooleanConstructor | ArrayConstructor} AnyValueConstructor + */ + +class IEntity { + + /** @type {AttributeDeclarations} */ + static attributes = {} + static defaultAttribute = { + showDefault: true, + nullable: false, + ignored: false, + serialized: false, + expected: false, + } + + constructor(values = {}, suppressWarns = false) { + /** + * @param {Object} target + * @param {Object} attributes + * @param {Object} values + * @param {String} prefix + */ + const defineAllAttributes = (target, attributes, values = {}, prefix = "") => { + const valuesNames = Object.keys(values); + const attributesNames = Object.keys(attributes); + const allAttributesNames = Utility.mergeArrays(attributesNames, valuesNames); + for (let attributeName of allAttributesNames) { + let value = Utility.objectGet(values, [attributeName]); + /** @type {AttributeInformation} */ + let attribute = attributes[attributeName]; + + if (attribute instanceof SubAttributesDeclaration) { + target[attributeName] = {}; + defineAllAttributes( + target[attributeName], + attribute.attributes, + values[attributeName], + attributeName + "." + ); + continue + } + + if (!suppressWarns) { + if (!(attributeName in attributes)) { + console.warn( + `UEBlueprint: Attribute ${prefix}${attributeName} in the serialized data is not defined in ` + + `${this.constructor.name}.attributes` + ); + } else if ( + valuesNames.length > 0 + && !(attributeName in values) + && !(!attribute.showDefault || attribute.ignored) + ) { + console.warn( + `UEBlueprint: ${this.constructor.name} will add attribute ${prefix}${attributeName} not ` + + "defined in the serialized data" + ); + } + } + + if (!attribute) { + // Remember attributeName can come from the values and be not defined in the attributes + target[attributeName] = value; + continue + } + + let defaultValue = attribute.value; + let defaultType = attribute.type; + if (attribute.serialized && defaultType instanceof Function) { + // If the attribute is serialized, the type must contain a function providing the type + defaultType = /** @type {TypeSupplier} */(defaultType)(this); + } + if (defaultType instanceof Array) { + defaultType = Array; + } + if (defaultValue instanceof Function) { + defaultValue = defaultValue(this); + } + if (defaultType === undefined) { + defaultType = Utility.getType(defaultValue); + } + const assignAttribute = !attribute.predicate + ? v => target[attributeName] = v + : v => { + Object.defineProperties(target, { + ["#" + attributeName]: { + writable: true, + enumerable: false, + }, + [attributeName]: { + enumerable: true, + get() { + return this["#" + attributeName] + }, + set(v) { + if (!attribute.predicate?.(v)) { + console.warn( + `UEBlueprint: Tried to assign attribute ${prefix}${attributeName} to ` + + `${this.constructor.name} not satisfying the predicate` + ); + return + } + this["#" + attributeName] = v; + } + }, + }); + this[attributeName] = v; + }; + + if (value !== undefined) { + // Remember value can still be null + if (value?.constructor === String && attribute.serialized && defaultType !== String) { + value = SerializerFactory + .getSerializer(/** @type {AnyValueConstructor<*>} */(defaultType)) + .deserialize(/** @type {String} */(value)); + } + assignAttribute(Utility.sanitize(value, /** @type {AnyValueConstructor<*>} */(defaultType))); + continue // We have a value, need nothing more + } + if (defaultType instanceof UnionType) { + if (defaultValue != undefined) { + defaultType = defaultType.types.find( + type => defaultValue instanceof type || defaultValue.constructor == type + ) ?? defaultType.getFirstType(); + } else { + defaultType = defaultType.getFirstType(); + } + } + if (defaultValue === undefined) { + defaultValue = Utility.sanitize(new /** @type {AnyValueConstructor<*>} */(defaultType)()); + } + if (!attribute.showDefault) { + assignAttribute(undefined); // Declare undefined to preserve the order of attributes + continue + } + if (attribute.serialized) { + if (defaultType !== String && defaultValue.constructor === String) { + defaultValue = SerializerFactory + .getSerializer(/** @type {AnyValueConstructor<*>} */(defaultType)) + .deserialize(defaultValue); + } + } + assignAttribute(Utility.sanitize( + /** @type {AnyValue} */(defaultValue), + /** @type {AnyValueConstructor} */(defaultType) + )); + } + }; + const attributes = /** @type {typeof IEntity} */(this.constructor).attributes; + if (values.constructor !== Object && Object.keys(attributes).length === 1) { + // Where there is just one attribute, option can be the value of that attribute + values = { + [Object.keys(attributes)[0]]: values + }; + } + defineAllAttributes(this, attributes, values); + } + + /** @param {AttributeDeclarations} attributes */ + static cleanupAttributes(attributes, prefix = "") { + for (const attributeName in attributes) { + if (attributes[attributeName] instanceof SubAttributesDeclaration) { + this.cleanupAttributes( + /** @type {SubAttributesDeclaration} */(attributes[attributeName]).attributes, + prefix + "." + attributeName + ); + continue + } + if (attributes[attributeName].constructor !== Object) { + attributes[attributeName] = { + value: attributes[attributeName], + }; + } + const attribute = /** @type {AttributeInformation} */(attributes[attributeName]); + if (attribute.type === undefined && !(attribute.value instanceof Function)) { + attribute.type = Utility.getType(attribute.value); + } + attributes[attributeName] = { + ...IEntity.defaultAttribute, + ...attribute, + }; + if (attribute.value === undefined && attribute.type === undefined) { + throw new Error( + `UEBlueprint: Expected either "type" or "value" property in ${this.name} attribute ${prefix}` + + attributeName + ) + } + if (attribute.value === null) { + attributes[attributeName].nullable = true; + } + } + } + + static isValueOfType(value, type) { + return value != null && (value instanceof type || value.constructor === type) + } + + static expectsAllKeys() { + return !Object.values(this.attributes) + .filter(/** @param {AttributeInformation} attribute */attribute => !attribute.ignored) + .some(/** @param {AttributeInformation} attribute */attribute => !attribute.expected) + } + + unexpectedKeys() { + return Object.keys(this).length + - Object.keys(/** @type {typeof IEntity} */(this.constructor).attributes).length + } + + /** @param {IEntity} other */ + equals(other) { + const thisKeys = Object.keys(this); + const otherKeys = Object.keys(this); + if (thisKeys.length != otherKeys.length) { + return false + } + for (const key of thisKeys) { + if (this[key] instanceof IEntity && !this[key].equals(other[key])) { + return false + } else if (!Utility.equals(this[key], other[key])) { + return false + } + } + return true + } +} + class IntegerEntity extends IEntity { static attributes = { @@ -1124,8 +1126,8 @@ class IntegerEntity extends IEntity { toString() { return this.value.toString() } -} - +} + class ByteEntity extends IntegerEntity { static attributes = { @@ -1143,11 +1145,59 @@ class ByteEntity extends IntegerEntity { constructor(values = 0) { super(values); } -} +} + +class SymbolEntity extends IEntity { + + static attributes = { + value: "", + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {String} */ this.value; + } + + valueOf() { + return this.value + } + + toString() { + return this.value + } +} + +class EnumEntity extends SymbolEntity { -class SymbolEntity extends IEntity { +} + +class InvariantTextEntity extends IEntity { + + static lookbehind = "INVTEXT" + static attributes = { + value: "", + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {String} */ this.value; + } +} + +class LocalizedTextEntity extends IEntity { + static lookbehind = "NSLOCTEXT" static attributes = { + namespace: "", + key: "", value: "", } @@ -1157,22 +1207,35 @@ class SymbolEntity extends IEntity { constructor(values) { super(values); + /** @type {String} */ this.namespace; + /** @type {String} */ this.key; /** @type {String} */ this.value; } - valueOf() { - return this.value - } - toString() { - return this.value + return Utility.capitalFirstLetter(this.value) } -} - -class EnumEntity extends SymbolEntity { - -} - +} + +class FormatTextEntity extends IEntity { + + static lookbehind = "LOCGEN_FORMAT_NAMED" + static attributes = { + value: { + type: [new UnionType(LocalizedTextEntity, InvariantTextEntity, FormatTextEntity)] + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {String} */ this.value; + } +} + class GuidEntity extends IEntity { static attributes = { @@ -1210,8 +1273,8 @@ class GuidEntity extends IEntity { toString() { return this.value } -} - +} + class ObjectReferenceEntity extends IEntity { static attributes = { @@ -1237,8 +1300,8 @@ class ObjectReferenceEntity extends IEntity { getName() { return this.path.match(/[^\.\/]+$/)?.[0] ?? "" } -} - +} + class FunctionReferenceEntity extends IEntity { static attributes = { @@ -1265,37 +1328,37 @@ class FunctionReferenceEntity extends IEntity { /** @type {ObjectReferenceEntity} */ this.MemberParent; /** @type {String} */ this.MemberName; } -} - -class IdentifierEntity extends IEntity { - - static attributes = { - value: "", - } - - static { - this.cleanupAttributes(this.attributes); - } - - static attributeConverter = { - fromAttribute: (value, type) => new IdentifierEntity(value), - toAttribute: (value, type) => value.toString() - } - - constructor(values) { - super(values); - /** @type {String} */ this.value; - } - - valueOf() { - return this.value - } - - toString() { - return this.value - } -} - +} + +class IdentifierEntity extends IEntity { + + static attributes = { + value: "", + } + + static { + this.cleanupAttributes(this.attributes); + } + + static attributeConverter = { + fromAttribute: (value, type) => new IdentifierEntity(value), + toAttribute: (value, type) => value.toString() + } + + constructor(values) { + super(values); + /** @type {String} */ this.value; + } + + valueOf() { + return this.value + } + + toString() { + return this.value + } +} + class Integer64Entity extends IEntity { static attributes = { @@ -1322,349 +1385,307 @@ class Integer64Entity extends IEntity { toString() { return this.value.toString() } -} - -class InvariantTextEntity extends IEntity { - - static lookbehind = "INVTEXT" - static attributes = { - value: "", - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {String} */ this.value; - } -} - -class KeyBindingEntity extends IEntity { - - static attributes = { - ActionName: "", - bShift: false, - bCtrl: false, - bAlt: false, - bCmd: false, - Key: { - type: IdentifierEntity - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values = {}) { - values.ActionName = values.ActionName ?? ""; - values.bShift = values.bShift ?? false; - values.bCtrl = values.bCtrl ?? false; - values.bAlt = values.bAlt ?? false; - values.bCmd = values.bCmd ?? false; - super(values); - /** @type {String} */ this.ActionName; - /** @type {Boolean} */ this.bShift; - /** @type {Boolean} */ this.bCtrl; - /** @type {Boolean} */ this.bAlt; - /** @type {Boolean} */ this.bCmd; - /** @type {IdentifierEntity} */ this.Key; - } -} - -class RealUnitEntity extends IEntity { - - static attributes = { - value: 0, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values = 0) { - super(values); - this.value = Utility.clamp(this.value, 0, 1); - } - - valueOf() { - return this.value - } - - toString() { - return this.value.toFixed(6) - } -} - -class LinearColorEntity extends IEntity { - - static attributes = { - R: { - type: RealUnitEntity, - expected: true, - }, - G: { - type: RealUnitEntity, - expected: true, - }, - B: { - type: RealUnitEntity, - expected: true, - }, - A: { - type: RealUnitEntity, - value: () => new RealUnitEntity(1), - }, - H: { - type: RealUnitEntity, - showDefault: true, - ignored: true, - }, - S: { - type: RealUnitEntity, - showDefault: true, - ignored: true, - }, - V: { - type: RealUnitEntity, - showDefault: true, - ignored: true, - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - /** @param {Number} x */ - static linearToSRGB(x) { - if (x <= 0) { - return 0 - } else if (x >= 1) { - return 1 - } else if (x < 0.0031308) { - return x * 12.92 - } else { - return Math.pow(x, 1 / 2.4) * 1.055 - 0.055 - } - } - - /** @param {Number} x */ - static sRGBtoLinear(x) { - if (x <= 0) { - return 0 - } else if (x >= 1) { - return 1 - } else if (x < 0.04045) { - return x / 12.92 - } else { - return Math.pow((x + 0.055) / 1.055, 2.4) - } - } - - static getWhite() { - return new LinearColorEntity({ - R: 1, - G: 1, - B: 1, - }) - } - - constructor(values) { - if (values instanceof Array) { - values = { - R: values[0] ?? 0, - G: values[1] ?? 0, - B: values[2] ?? 0, - A: values[3] ?? 1, - }; - } - super(values); - /** @type {RealUnitEntity} */ this.R; - /** @type {RealUnitEntity} */ this.G; - /** @type {RealUnitEntity} */ this.B; - /** @type {RealUnitEntity} */ this.A; - /** @type {RealUnitEntity} */ this.H; - /** @type {RealUnitEntity} */ this.S; - /** @type {RealUnitEntity} */ this.V; - this.#updateHSV(); - } - - #updateHSV() { - const r = this.R.value; - const g = this.G.value; - const b = this.B.value; - if (Utility.approximatelyEqual(r, g) && Utility.approximatelyEqual(r, b) && Utility.approximatelyEqual(g, b)) { - this.S.value = 0; - this.V.value = r; - return - } - const max = Math.max(r, g, b); - const min = Math.min(r, g, b); - const d = max - min; - let h; - switch (max) { - case min: - h = 0; - break - case r: - h = (g - b) / d + (g < b ? 6 : 0); - break - case g: - h = (b - r) / d + 2; - break - case b: - h = (r - g) / d + 4; - break - } - h /= 6; - this.H.value = h; - this.S.value = max == 0 ? 0 : d / max; - this.V.value = max; - } - - /** - * @param {Number} r - * @param {Number} g - * @param {Number} b - * @param {Number} a - */ - setFromRGBA(r, g, b, a = 1) { - this.R.value = r; - this.G.value = g; - this.B.value = b; - this.A.value = a; - this.#updateHSV(); - } - - /** - * @param {Number} h - * @param {Number} s - * @param {Number} v - * @param {Number} a - */ - setFromHSVA(h, s, v, a = 1) { - const i = Math.floor(h * 6); - const f = h * 6 - i; - const p = v * (1 - s); - const q = v * (1 - f * s); - const t = v * (1 - (1 - f) * s); - const values = [v, q, p, p, t, v]; - const [r, g, b] = [values[i % 6], values[(i + 4) % 6], values[(i + 2) % 6]]; - this.R.value = r; - this.G.value = g; - this.B.value = b; - this.A.value = a; - this.H.value = h; - this.S.value = s; - this.V.value = v; - } - - /** - * @param {Number} x - * @param {Number} y - * @param {Number} v - * @param {Number} a - */ - setFromWheelLocation(x, y, v, a) { - const [r, theta] = Utility.getPolarCoordinates(x, y, true); - this.setFromHSVA(1 - theta / (2 * Math.PI), r, v, a); - } - - toRGBA() { - return [ - Math.round(this.R.value * 255), - Math.round(this.G.value * 255), - Math.round(this.B.value * 255), - Math.round(this.A.value * 255), - ] - } - - toSRGBA() { - return [ - Math.round(LinearColorEntity.linearToSRGB(this.R.value) * 255), - Math.round(LinearColorEntity.linearToSRGB(this.G.value) * 255), - Math.round(LinearColorEntity.linearToSRGB(this.B.value) * 255), - Math.round(this.A.value * 255), - ] - } - - toRGBAString() { - return this - .toRGBA() - .map(v => v.toString(16).toUpperCase().padStart(2, "0")) - .join("") - } - - toSRGBAString() { - return this - .toSRGBA() - .map(v => v.toString(16).toUpperCase().padStart(2, "0")) - .join("") - } - - toHSVA() { - return [this.H.value, this.S.value, this.V.value, this.A.value] - } - - toNumber() { - return ( - Math.round(this.R.value * 0xff) << 24) - + (Math.round(this.G.value * 0xff) << 16) - + (Math.round(this.B.value * 0xff) << 8) - + Math.round(this.A.value * 0xff) - } - - /** @param {Number} number */ - setFromRGBANumber(number) { - this.A.value = (number & 0xff) / 0xff; - this.B.value = ((number >> 8) & 0xff) / 0xff; - this.G.value = ((number >> 16) & 0xff) / 0xff; - this.R.value = ((number >> 24) & 0xff) / 0xff; - this.#updateHSV(); - } - - /** @param {Number} number */ - setFromSRGBANumber(number) { - this.A.value = (number & 0xff) / 0xff; - this.B.value = LinearColorEntity.sRGBtoLinear(((number >> 8) & 0xff) / 0xff); - this.G.value = LinearColorEntity.sRGBtoLinear(((number >> 16) & 0xff) / 0xff); - this.R.value = LinearColorEntity.sRGBtoLinear(((number >> 24) & 0xff) / 0xff); - this.#updateHSV(); - } - - toString() { - return Utility.printLinearColor(this) - } -} - -class LocalizedTextEntity extends IEntity { - - static lookbehind = "NSLOCTEXT" - static attributes = { - namespace: "", - key: "", - value: "", - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {String} */ this.namespace; - /** @type {String} */ this.key; - /** @type {String} */ this.value; - } - - toString() { - return Utility.capitalFirstLetter(this.value) - } -} - +} + +class KeyBindingEntity extends IEntity { + + static attributes = { + ActionName: "", + bShift: false, + bCtrl: false, + bAlt: false, + bCmd: false, + Key: { + type: IdentifierEntity + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values = {}) { + values.ActionName = values.ActionName ?? ""; + values.bShift = values.bShift ?? false; + values.bCtrl = values.bCtrl ?? false; + values.bAlt = values.bAlt ?? false; + values.bCmd = values.bCmd ?? false; + super(values); + /** @type {String} */ this.ActionName; + /** @type {Boolean} */ this.bShift; + /** @type {Boolean} */ this.bCtrl; + /** @type {Boolean} */ this.bAlt; + /** @type {Boolean} */ this.bCmd; + /** @type {IdentifierEntity} */ this.Key; + } +} + +class RealUnitEntity extends IEntity { + + static attributes = { + value: 0, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values = 0) { + super(values); + this.value = Utility.clamp(this.value, 0, 1); + } + + valueOf() { + return this.value + } + + toString() { + return this.value.toFixed(6) + } +} + +class LinearColorEntity extends IEntity { + + static attributes = { + R: { + type: RealUnitEntity, + expected: true, + }, + G: { + type: RealUnitEntity, + expected: true, + }, + B: { + type: RealUnitEntity, + expected: true, + }, + A: { + type: RealUnitEntity, + value: () => new RealUnitEntity(1), + }, + H: { + type: RealUnitEntity, + showDefault: true, + ignored: true, + }, + S: { + type: RealUnitEntity, + showDefault: true, + ignored: true, + }, + V: { + type: RealUnitEntity, + showDefault: true, + ignored: true, + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + /** @param {Number} x */ + static linearToSRGB(x) { + if (x <= 0) { + return 0 + } else if (x >= 1) { + return 1 + } else if (x < 0.0031308) { + return x * 12.92 + } else { + return Math.pow(x, 1 / 2.4) * 1.055 - 0.055 + } + } + + /** @param {Number} x */ + static sRGBtoLinear(x) { + if (x <= 0) { + return 0 + } else if (x >= 1) { + return 1 + } else if (x < 0.04045) { + return x / 12.92 + } else { + return Math.pow((x + 0.055) / 1.055, 2.4) + } + } + + static getWhite() { + return new LinearColorEntity({ + R: 1, + G: 1, + B: 1, + }) + } + + constructor(values) { + if (values instanceof Array) { + values = { + R: values[0] ?? 0, + G: values[1] ?? 0, + B: values[2] ?? 0, + A: values[3] ?? 1, + }; + } + super(values); + /** @type {RealUnitEntity} */ this.R; + /** @type {RealUnitEntity} */ this.G; + /** @type {RealUnitEntity} */ this.B; + /** @type {RealUnitEntity} */ this.A; + /** @type {RealUnitEntity} */ this.H; + /** @type {RealUnitEntity} */ this.S; + /** @type {RealUnitEntity} */ this.V; + this.#updateHSV(); + } + + #updateHSV() { + const r = this.R.value; + const g = this.G.value; + const b = this.B.value; + if (Utility.approximatelyEqual(r, g) && Utility.approximatelyEqual(r, b) && Utility.approximatelyEqual(g, b)) { + this.S.value = 0; + this.V.value = r; + return + } + const max = Math.max(r, g, b); + const min = Math.min(r, g, b); + const d = max - min; + let h; + switch (max) { + case min: + h = 0; + break + case r: + h = (g - b) / d + (g < b ? 6 : 0); + break + case g: + h = (b - r) / d + 2; + break + case b: + h = (r - g) / d + 4; + break + } + h /= 6; + this.H.value = h; + this.S.value = max == 0 ? 0 : d / max; + this.V.value = max; + } + + /** + * @param {Number} r + * @param {Number} g + * @param {Number} b + * @param {Number} a + */ + setFromRGBA(r, g, b, a = 1) { + this.R.value = r; + this.G.value = g; + this.B.value = b; + this.A.value = a; + this.#updateHSV(); + } + + /** + * @param {Number} h + * @param {Number} s + * @param {Number} v + * @param {Number} a + */ + setFromHSVA(h, s, v, a = 1) { + const i = Math.floor(h * 6); + const f = h * 6 - i; + const p = v * (1 - s); + const q = v * (1 - f * s); + const t = v * (1 - (1 - f) * s); + const values = [v, q, p, p, t, v]; + const [r, g, b] = [values[i % 6], values[(i + 4) % 6], values[(i + 2) % 6]]; + this.R.value = r; + this.G.value = g; + this.B.value = b; + this.A.value = a; + this.H.value = h; + this.S.value = s; + this.V.value = v; + } + + /** + * @param {Number} x + * @param {Number} y + * @param {Number} v + * @param {Number} a + */ + setFromWheelLocation(x, y, v, a) { + const [r, theta] = Utility.getPolarCoordinates(x, y, true); + this.setFromHSVA(1 - theta / (2 * Math.PI), r, v, a); + } + + toRGBA() { + return [ + Math.round(this.R.value * 255), + Math.round(this.G.value * 255), + Math.round(this.B.value * 255), + Math.round(this.A.value * 255), + ] + } + + toSRGBA() { + return [ + Math.round(LinearColorEntity.linearToSRGB(this.R.value) * 255), + Math.round(LinearColorEntity.linearToSRGB(this.G.value) * 255), + Math.round(LinearColorEntity.linearToSRGB(this.B.value) * 255), + Math.round(this.A.value * 255), + ] + } + + toRGBAString() { + return this + .toRGBA() + .map(v => v.toString(16).toUpperCase().padStart(2, "0")) + .join("") + } + + toSRGBAString() { + return this + .toSRGBA() + .map(v => v.toString(16).toUpperCase().padStart(2, "0")) + .join("") + } + + toHSVA() { + return [this.H.value, this.S.value, this.V.value, this.A.value] + } + + toNumber() { + return ( + Math.round(this.R.value * 0xff) << 24) + + (Math.round(this.G.value * 0xff) << 16) + + (Math.round(this.B.value * 0xff) << 8) + + Math.round(this.A.value * 0xff) + } + + /** @param {Number} number */ + setFromRGBANumber(number) { + this.A.value = (number & 0xff) / 0xff; + this.B.value = ((number >> 8) & 0xff) / 0xff; + this.G.value = ((number >> 16) & 0xff) / 0xff; + this.R.value = ((number >> 24) & 0xff) / 0xff; + this.#updateHSV(); + } + + /** @param {Number} number */ + setFromSRGBANumber(number) { + this.A.value = (number & 0xff) / 0xff; + this.B.value = LinearColorEntity.sRGBtoLinear(((number >> 8) & 0xff) / 0xff); + this.G.value = LinearColorEntity.sRGBtoLinear(((number >> 16) & 0xff) / 0xff); + this.R.value = LinearColorEntity.sRGBtoLinear(((number >> 24) & 0xff) / 0xff); + this.#updateHSV(); + } + + toString() { + return Utility.printLinearColor(this) + } +} + class MacroGraphReferenceEntity extends IEntity { static attributes = { @@ -1694,8 +1715,8 @@ class MacroGraphReferenceEntity extends IEntity { const colonIndex = this.MacroGraph.path.search(":"); return this.MacroGraph.path.substring(colonIndex + 1) } -} - +} + class PathSymbolEntity extends IEntity { static attributes = { @@ -1718,8 +1739,8 @@ class PathSymbolEntity extends IEntity { toString() { return this.value } -} - +} + class PinReferenceEntity extends IEntity { static attributes = { @@ -1740,147 +1761,147 @@ class PinReferenceEntity extends IEntity { /** @type {PathSymbolEntity} */ this.objectName; /** @type {GuidEntity} */ this.pinGuid; } -} - -class PinTypeEntity extends IEntity { - - static attributes = { - TerminalCategory: { - value: "", - showDefault: false, - }, - TerminalSubCategory: { - value: "", - showDefault: false, - }, - bTerminalIsConst: { - value: false, - showDefault: false, - }, - bTerminalIsWeakPointer: { - value: false, - showDefault: false, - }, - bTerminalIsUObjectWrapper: { - value: false, - showDefault: false, - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {String} */ this.TerminalCategory; - /** @type {String} */ this.TerminalSubCategory; - /** @type {Boolean} */ this.bTerminalIsConst; - /** @type {Boolean} */ this.bTerminalIsWeakPointer; - /** @type {Boolean} */ this.bTerminalIsUObjectWrapper; - } -} - -class RotatorEntity extends IEntity { - - static attributes = { - R: { - value: 0, - }, - P: { - value: 0, - }, - Y: { - value: 0, - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {Number} */ this.R; - /** @type {Number} */ this.P; - /** @type {Number} */ this.Y; - } - - getRoll() { - return this.R - } - - getPitch() { - return this.P - } - - getYaw() { - return this.Y - } -} - -class SimpleSerializationRotatorEntity extends RotatorEntity { -} - -class Vector2DEntity extends IEntity { - - static attributes = { - X: { - value: 0, - expected: true, - }, - Y: { - value: 0, - expected: true, - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {Number} */ this.X; - /** @type {Number} */ this.Y; - } -} - -class SimpleSerializationVector2DEntity extends Vector2DEntity { -} - -class VectorEntity extends IEntity { - - static attributes = { - X: { - value: 0, - expected: true, - }, - Y: { - value: 0, - expected: true, - }, - Z: { - value: 0, - expected: true, - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {Number} */ this.X; - /** @type {Number} */ this.Y; - /** @type {Number} */ this.Z; - } -} - -class SimpleSerializationVectorEntity extends VectorEntity { -} - +} + +class PinTypeEntity extends IEntity { + + static attributes = { + TerminalCategory: { + value: "", + showDefault: false, + }, + TerminalSubCategory: { + value: "", + showDefault: false, + }, + bTerminalIsConst: { + value: false, + showDefault: false, + }, + bTerminalIsWeakPointer: { + value: false, + showDefault: false, + }, + bTerminalIsUObjectWrapper: { + value: false, + showDefault: false, + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {String} */ this.TerminalCategory; + /** @type {String} */ this.TerminalSubCategory; + /** @type {Boolean} */ this.bTerminalIsConst; + /** @type {Boolean} */ this.bTerminalIsWeakPointer; + /** @type {Boolean} */ this.bTerminalIsUObjectWrapper; + } +} + +class RotatorEntity extends IEntity { + + static attributes = { + R: { + value: 0, + }, + P: { + value: 0, + }, + Y: { + value: 0, + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {Number} */ this.R; + /** @type {Number} */ this.P; + /** @type {Number} */ this.Y; + } + + getRoll() { + return this.R + } + + getPitch() { + return this.P + } + + getYaw() { + return this.Y + } +} + +class SimpleSerializationRotatorEntity extends RotatorEntity { +} + +class Vector2DEntity extends IEntity { + + static attributes = { + X: { + value: 0, + expected: true, + }, + Y: { + value: 0, + expected: true, + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {Number} */ this.X; + /** @type {Number} */ this.Y; + } +} + +class SimpleSerializationVector2DEntity extends Vector2DEntity { +} + +class VectorEntity extends IEntity { + + static attributes = { + X: { + value: 0, + expected: true, + }, + Y: { + value: 0, + expected: true, + }, + Z: { + value: 0, + expected: true, + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {Number} */ this.X; + /** @type {Number} */ this.Y; + /** @type {Number} */ this.Z; + } +} + +class SimpleSerializationVectorEntity extends VectorEntity { +} + /** * @typedef {import("./IEntity").AnyValue} AnyValue * @typedef {import("lit").CSSResult} CSSResult @@ -1916,7 +1937,7 @@ class PinEntity extends IEntity { }, PinName: "", PinFriendlyName: { - type: new UnionType(LocalizedTextEntity, String), + type: new UnionType(LocalizedTextEntity, FormatTextEntity, String), showDefault: false, }, PinToolTip: { @@ -2147,300 +2168,308 @@ class PinEntity extends IEntity { ?? Configuration.pinColor[this.PinType.PinCategory] ?? Configuration.pinColor["default"] } -} - -class SVGIcon { - - static array = y` - - - - - - - - - - - - ` - - static branchNode = y` - - - - - - - ` - - static breakStruct = y` - - - - - - ` - - static cast = y` - - - - - - - - - ` - - static close = y` - - - - - ` - - static correct = y` - - - - ` - - static delegate = y` - - - - ` - - static doN = y` - - - - - ` - - static enum = y` - - - - - ` - - static event = y` - - - - - - - ` - - static execPin = y` - - - - ` - - static expandIcon = y` - - - - ` - - static forEachLoop = y` - - - - - - - - - ` - - static functionSymbol = y` - - - - ` - - static gamepad = y` - - - - ` - - static genericPin = y` - - - - - ` - - static keyboard = y` - - - - ` - - static loop = y` - - - - - - - - - - - - ` - - static macro = y` - - - - ` - - static map = y` - - - - - - - - - ` - - static makeArray = y` - - - - - - - - - - - - - - ` - - static makeMap = y` - - - - - - - - - - - ` - - static makeSet = y` - - - - - - - - - ` - - static makeStruct = y` - - - - - - ` - - static mouse = y` - - - - - ` - - static questionMark = y` - - - - - ` - - static referencePin = y` - - - - ` - - static reject = y` - - - - - ` - - static set = y` - - - - - - - ` - - static select = y` - - - - - - - - - ` - - static sequence = y` - - - - - - - - - - ` - - static touchpad = y` - - - - - ` -} - +} + +class SVGIcon { + + static array = y` + + + + + + + + + + + + ` + + static branchNode = y` + + + + + + + ` + + static breakStruct = y` + + + + + + ` + + static cast = y` + + + + + + + + + ` + + static close = y` + + + + + ` + + static correct = y` + + + + ` + + static delegate = y` + + + + ` + + static doN = y` + + + + + ` + + static doOnce = y` + + + + + + ` + + static enum = y` + + + + + ` + + static event = y` + + + + + + + ` + + static execPin = y` + + + + ` + + static expandIcon = y` + + + + ` + + static forEachLoop = y` + + + + + + + + + ` + + static functionSymbol = y` + + + + ` + + static gamepad = y` + + + + ` + + static genericPin = y` + + + + + ` + + static keyboard = y` + + + + ` + + static loop = y` + + + + + + + + + + + + ` + + static macro = y` + + + + ` + + static map = y` + + + + + + + + + ` + + static makeArray = y` + + + + + + + + + + + + + + ` + + static makeMap = y` + + + + + + + + + + + ` + + static makeSet = y` + + + + + + + + + ` + + static makeStruct = y` + + + + + + ` + + static mouse = y` + + + + + ` + + static questionMark = y` + + + + + ` + + static referencePin = y` + + + + ` + + static reject = y` + + + + + ` + + static set = y` + + + + + + + ` + + static select = y` + + + + + + + + + ` + + static sequence = y` + + + + + + + + + + ` + + static touchpad = y` + + + + + ` +} + class VariableReferenceEntity extends IEntity { static attributes = { @@ -2468,8 +2497,8 @@ class VariableReferenceEntity extends IEntity { /** @type {GuidEntity} */ this.GuidEntity; /** @type {Boolean} */ this.bSelfContext; } -} - +} + class ObjectEntity extends IEntity { static attributes = { @@ -2761,7 +2790,7 @@ class ObjectEntity extends IEntity { getType() { let classValue = this.getClass(); - if (classValue === Configuration.nodeType.macro) { + if (this.MacroGraphReference?.MacroGraph?.path) { return this.MacroGraphReference.MacroGraph.path } return classValue @@ -2864,7 +2893,7 @@ class ObjectEntity extends IEntity { || nodeClass.includes("Debug", Math.max(0, nodeClass.lastIndexOf("."))) } - hasHIDAttribute() { + getHIDAttribute() { return this.InputKey ?? this.AxisKey ?? this.InputAxisKey } @@ -2880,15 +2909,15 @@ class ObjectEntity extends IEntity { if (!this.TargetType) { return "Bad cast node" // Target type not found } - return `Cast To ${this.TargetType.getName()}` + return `Cast To ${this.TargetType?.getName()}` case Configuration.nodeType.enumLiteral: - return `Literal enum ${this.Enum.getName()}` + return `Literal enum ${this.Enum?.getName()}` case Configuration.nodeType.event: return `Event ${(this.EventReference?.MemberName ?? "").replace(/^Receive/, "")}` case Configuration.nodeType.executionSequence: return "Sequence" case Configuration.nodeType.forEachElementInEnum: - return `For Each ${this.Enum.getName()}` + return `For Each ${this.Enum?.getName()}` case Configuration.nodeType.forEachLoopWithBreak: return "For Each Loop with Break" case Configuration.nodeType.ifThenElse: @@ -2898,7 +2927,7 @@ class ObjectEntity extends IEntity { case Configuration.nodeType.variableSet: return "SET" } - const keyNameSymbol = this.hasHIDAttribute(); + const keyNameSymbol = this.getHIDAttribute(); if (keyNameSymbol) { const keyName = keyNameSymbol.toString(); let title = ObjectEntity.keyName(keyName) ?? Utility.formatStringName(keyName); @@ -2910,7 +2939,7 @@ class ObjectEntity extends IEntity { return title } if (this.getClass() === Configuration.nodeType.macro) { - return Utility.formatStringName(this.MacroGraphReference.getMacroName()) + return Utility.formatStringName(this.MacroGraphReference?.getMacroName()) } let memberName = this.FunctionReference?.MemberName; if (memberName) { @@ -2997,6 +3026,7 @@ class ObjectEntity extends IEntity { case Configuration.nodeType.executionSequence: case Configuration.nodeType.ifThenElse: case Configuration.nodeType.macro: + case Configuration.nodeType.multiGate: return Configuration.nodeColors.gray case Configuration.nodeType.dynamicCast: return Configuration.nodeColors.turquoise @@ -3014,22 +3044,27 @@ class ObjectEntity extends IEntity { switch (this.getType()) { case Configuration.nodeType.customEvent: return SVGIcon.event case Configuration.nodeType.doN: return SVGIcon.doN + case Configuration.nodeType.doOnce: return SVGIcon.doOnce case Configuration.nodeType.dynamicCast: return SVGIcon.cast case Configuration.nodeType.enumLiteral: return SVGIcon.enum case Configuration.nodeType.event: return SVGIcon.event - case Configuration.nodeType.executionSequence: return SVGIcon.sequence - case Configuration.nodeType.forEachElementInEnum: return SVGIcon.loop - case Configuration.nodeType.forEachLoop: return SVGIcon.forEachLoop - case Configuration.nodeType.forEachLoopWithBreak: return SVGIcon.forEachLoop - case Configuration.nodeType.forLoop: return SVGIcon.loop - case Configuration.nodeType.forLoopWithBreak: return SVGIcon.loop + case Configuration.nodeType.executionSequence: + case Configuration.nodeType.multiGate: + return SVGIcon.sequence + case Configuration.nodeType.forEachElementInEnum: + case Configuration.nodeType.forLoop: + case Configuration.nodeType.forLoopWithBreak: + case Configuration.nodeType.whileLoop: + return SVGIcon.loop + case Configuration.nodeType.forEachLoop: + case Configuration.nodeType.forEachLoopWithBreak: + return SVGIcon.forEachLoop case Configuration.nodeType.ifThenElse: return SVGIcon.branchNode case Configuration.nodeType.isValid: return SVGIcon.questionMark case Configuration.nodeType.makeArray: return SVGIcon.makeArray case Configuration.nodeType.makeMap: return SVGIcon.makeMap case Configuration.nodeType.makeSet: return SVGIcon.makeSet case Configuration.nodeType.select: return SVGIcon.select - case Configuration.nodeType.whileLoop: return SVGIcon.loop } if (this.nodeDisplayName().startsWith("Break")) { return SVGIcon.breakStruct @@ -3037,7 +3072,7 @@ class ObjectEntity extends IEntity { if (this.getClass() === Configuration.nodeType.macro) { return SVGIcon.macro } - const hidValue = this.hasHIDAttribute()?.toString(); + const hidValue = this.getHIDAttribute()?.toString(); if (hidValue) { if (hidValue.includes("Mouse")) { return SVGIcon.mouse @@ -3056,43 +3091,43 @@ class ObjectEntity extends IEntity { } return SVGIcon.functionSymbol } -} - -var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {}; - -function getDefaultExportFromCjs (x) { - return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x; -} - -var parsimmon_umd_min = {exports: {}}; - -(function (module, exports) { -!function(n,t){module.exports=t();}("undefined"!=typeof self?self:commonjsGlobal,function(){return function(n){var t={};function r(e){if(t[e])return t[e].exports;var u=t[e]={i:e,l:!1,exports:{}};return n[e].call(u.exports,u,u.exports,r),u.l=!0,u.exports}return r.m=n,r.c=t,r.d=function(n,t,e){r.o(n,t)||Object.defineProperty(n,t,{configurable:!1,enumerable:!0,get:e});},r.r=function(n){Object.defineProperty(n,"__esModule",{value:!0});},r.n=function(n){var t=n&&n.__esModule?function(){return n.default}:function(){return n};return r.d(t,"a",t),t},r.o=function(n,t){return Object.prototype.hasOwnProperty.call(n,t)},r.p="",r(r.s=0)}([function(n,t,r){function e(n){if(!(this instanceof e))return new e(n);this._=n;}var u=e.prototype;function o(n,t){for(var r=0;r>7),buf:function(n){var t=i(function(n,t,r,e){return n.concat(r===e.length-1?Buffer.from([t,0]).readUInt16BE(0):e.readUInt16BE(r))},[],n);return Buffer.from(a(function(n){return (n<<1&65535)>>8},t))}(r.buf)};}),r}function c(){return "undefined"!=typeof Buffer}function s(){if(!c())throw new Error("Buffer global does not exist; please use webpack if you need to parse Buffers in the browser.")}function l(n){s();var t=i(function(n,t){return n+t},0,n);if(t%8!=0)throw new Error("The bits ["+n.join(", ")+"] add up to "+t+" which is not an even number of bytes; the total should be divisible by 8");var r,u=t/8,o=(r=function(n){return n>48},i(function(n,t){return n||(r(t)?t:n)},null,n));if(o)throw new Error(o+" bit range requested exceeds 48 bit (6 byte) Number max.");return new e(function(t,r){var e=u+r;return e>t.length?x(r,u.toString()+" bytes"):b(e,i(function(n,t){var r=f(t,n.buf);return {coll:n.coll.concat(r.v),buf:r.buf}},{coll:[],buf:t.slice(r,e)},n).coll)})}function h(n,t){return new e(function(r,e){return s(),e+t>r.length?x(e,t+" bytes for "+n):b(e+t,r.slice(e,e+t))})}function p(n,t){if("number"!=typeof(r=t)||Math.floor(r)!==r||t<0||t>6)throw new Error(n+" requires integer length in range [0, 6].");var r;}function d(n){return p("uintBE",n),h("uintBE("+n+")",n).map(function(t){return t.readUIntBE(0,n)})}function v(n){return p("uintLE",n),h("uintLE("+n+")",n).map(function(t){return t.readUIntLE(0,n)})}function g(n){return p("intBE",n),h("intBE("+n+")",n).map(function(t){return t.readIntBE(0,n)})}function m(n){return p("intLE",n),h("intLE("+n+")",n).map(function(t){return t.readIntLE(0,n)})}function y(n){return n instanceof e}function E(n){return "[object Array]"==={}.toString.call(n)}function w(n){return c()&&Buffer.isBuffer(n)}function b(n,t){return {status:!0,index:n,value:t,furthest:-1,expected:[]}}function x(n,t){return E(t)||(t=[t]),{status:!1,index:-1,value:null,furthest:n,expected:t}}function B(n,t){if(!t)return n;if(n.furthest>t.furthest)return n;var r=n.furthest===t.furthest?function(n,t){if(function(){if(void 0!==e._supportsSet)return e._supportsSet;var n="undefined"!=typeof Set;return e._supportsSet=n,n}()&&Array.from){for(var r=new Set(n),u=0;u=0;){if(i in r){e=r[i].line,0===o&&(o=r[i].lineStart);break}("\n"===n.charAt(i)||"\r"===n.charAt(i)&&"\n"!==n.charAt(i+1))&&(u++,0===o&&(o=i+1)),i--;}var a=e+u,f=t-o;return r[t]={line:a,lineStart:o},{offset:t,line:a+1,column:f+1}}function _(n){if(!y(n))throw new Error("not a parser: "+n)}function L(n,t){return "string"==typeof n?n.charAt(t):n[t]}function O(n){if("number"!=typeof n)throw new Error("not a number: "+n)}function k(n){if("function"!=typeof n)throw new Error("not a function: "+n)}function P(n){if("string"!=typeof n)throw new Error("not a string: "+n)}var q=2,A=3,I=8,F=5*I,M=4*I,z=" ";function R(n,t){return new Array(t+1).join(n)}function U(n,t,r){var e=t-n.length;return e<=0?n:R(r,e)+n}function W(n,t,r,e){return {from:n-t>0?n-t:0,to:n+r>e?e:n+r}}function D(n,t){var r,e,u,o,f,c=t.index,s=c.offset,l=1;if(s===n.length)return "Got the end of the input";if(w(n)){var h=s-s%I,p=s-h,d=W(h,F,M+I,n.length),v=a(function(n){return a(function(n){return U(n.toString(16),2,"0")},n)},function(n,t){var r=n.length,e=[],u=0;if(r<=t)return [n.slice()];for(var o=0;o=4&&(r+=1),l=2,u=a(function(n){return n.length<=4?n.join(" "):n.slice(0,4).join(" ")+" "+n.slice(4).join(" ")},v),(f=(8*(o.to>0?o.to-1:o.to)).toString(16).length)<2&&(f=2);}else {var g=n.split(/\r\n|[\n\r\u2028\u2029]/);r=c.column-1,e=c.line-1,o=W(e,q,A,g.length),u=g.slice(o.from,o.to),f=o.to.toString().length;}var m=e-o.from;return w(n)&&(f=(8*(o.to>0?o.to-1:o.to)).toString(16).length)<2&&(f=2),i(function(t,e,u){var i,a=u===m,c=a?"> ":z;return i=w(n)?U((8*(o.from+u)).toString(16),f,"0"):U((o.from+u+1).toString(),f," "),[].concat(t,[c+i+" | "+e],a?[z+R(" ",f)+" | "+U("",r," ")+R("^",l)]:[])},[],u).join("\n")}function N(n,t){return ["\n","-- PARSING FAILED "+R("-",50),"\n\n",D(n,t),"\n\n",(r=t.expected,1===r.length?"Expected:\n\n"+r[0]:"Expected one of the following: \n\n"+r.join(", ")),"\n"].join("");var r;}function G(n){return void 0!==n.flags?n.flags:[n.global?"g":"",n.ignoreCase?"i":"",n.multiline?"m":"",n.unicode?"u":"",n.sticky?"y":""].join("")}function C(){for(var n=[].slice.call(arguments),t=n.length,r=0;r=2?O(t):t=0;var r=function(n){return RegExp("^(?:"+n.source+")",G(n))}(n),u=""+n;return e(function(n,e){var o=r.exec(n.slice(e));if(o){if(0<=t&&t<=o.length){var i=o[0],a=o[t];return b(e+i.length,a)}return x(e,"valid match group (0 to "+o.length+") in "+u)}return x(e,u)})}function X(n){return e(function(t,r){return b(r,n)})}function Y(n){return e(function(t,r){return x(r,n)})}function Z(n){if(y(n))return e(function(t,r){var e=n._(t,r);return e.index=r,e.value="",e});if("string"==typeof n)return Z(K(n));if(n instanceof RegExp)return Z(Q(n));throw new Error("not a string, regexp, or parser: "+n)}function $(n){return _(n),e(function(t,r){var e=n._(t,r),u=t.slice(r,e.index);return e.status?x(r,'not "'+u+'"'):b(r,null)})}function nn(n){return k(n),e(function(t,r){var e=L(t,r);return r=n.length?x(t,"any character/byte"):b(t+1,L(n,t))}),on=e(function(n,t){return b(n.length,n.slice(t))}),an=e(function(n,t){return t=0}).desc(t)},e.optWhitespace=hn,e.Parser=e,e.range=function(n,t){return nn(function(r){return n<=r&&r<=t}).desc(n+"-"+t)},e.regex=Q,e.regexp=Q,e.sepBy=V,e.sepBy1=H,e.seq=C,e.seqMap=J,e.seqObj=function(){for(var n,t={},r=0,u=(n=arguments,Array.prototype.slice.call(n)),o=u.length,i=0;i255)throw new Error("Value specified to byte constructor ("+n+"=0x"+n.toString(16)+") is larger in value than a single byte.");var t=(n>15?"0x":"0x0")+n.toString(16);return e(function(r,e){var u=L(r,e);return u===n?b(e+1,u):x(e,t)})},buffer:function(n){return h("buffer",n).map(function(n){return Buffer.from(n)})},encodedString:function(n,t){return h("string",t).map(function(t){return t.toString(n)})},uintBE:d,uint8BE:d(1),uint16BE:d(2),uint32BE:d(4),uintLE:v,uint8LE:v(1),uint16LE:v(2),uint32LE:v(4),intBE:g,int8BE:g(1),int16BE:g(2),int32BE:g(4),intLE:m,int8LE:m(1),int16LE:m(2),int32LE:m(4),floatBE:h("floatBE",4).map(function(n){return n.readFloatBE(0)}),floatLE:h("floatLE",4).map(function(n){return n.readFloatLE(0)}),doubleBE:h("doubleBE",8).map(function(n){return n.readDoubleBE(0)}),doubleLE:h("doubleLE",8).map(function(n){return n.readDoubleLE(0)})},n.exports=e;}])}); -}(parsimmon_umd_min)); - -var Parsimmon = /*@__PURE__*/getDefaultExportFromCjs(parsimmon_umd_min.exports); - -class UnknownKeysEntity extends IEntity { - - static attributes = { - lookbehind: - { - value: "", - showDefault: false, - ignore: true, - }, - } - - static { - this.cleanupAttributes(this.attributes); - } - - constructor(values) { - super(values); - /** @type {String} */ this.lookbehind; - } -} - +} + +var commonjsGlobal = typeof globalThis !== 'undefined' ? globalThis : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : typeof self !== 'undefined' ? self : {}; + +function getDefaultExportFromCjs (x) { + return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x; +} + +var parsimmon_umd_min = {exports: {}}; + +(function (module, exports) { +!function(n,t){module.exports=t();}("undefined"!=typeof self?self:commonjsGlobal,function(){return function(n){var t={};function r(e){if(t[e])return t[e].exports;var u=t[e]={i:e,l:!1,exports:{}};return n[e].call(u.exports,u,u.exports,r),u.l=!0,u.exports}return r.m=n,r.c=t,r.d=function(n,t,e){r.o(n,t)||Object.defineProperty(n,t,{configurable:!1,enumerable:!0,get:e});},r.r=function(n){Object.defineProperty(n,"__esModule",{value:!0});},r.n=function(n){var t=n&&n.__esModule?function(){return n.default}:function(){return n};return r.d(t,"a",t),t},r.o=function(n,t){return Object.prototype.hasOwnProperty.call(n,t)},r.p="",r(r.s=0)}([function(n,t,r){function e(n){if(!(this instanceof e))return new e(n);this._=n;}var u=e.prototype;function o(n,t){for(var r=0;r>7),buf:function(n){var t=i(function(n,t,r,e){return n.concat(r===e.length-1?Buffer.from([t,0]).readUInt16BE(0):e.readUInt16BE(r))},[],n);return Buffer.from(a(function(n){return (n<<1&65535)>>8},t))}(r.buf)};}),r}function c(){return "undefined"!=typeof Buffer}function s(){if(!c())throw new Error("Buffer global does not exist; please use webpack if you need to parse Buffers in the browser.")}function l(n){s();var t=i(function(n,t){return n+t},0,n);if(t%8!=0)throw new Error("The bits ["+n.join(", ")+"] add up to "+t+" which is not an even number of bytes; the total should be divisible by 8");var r,u=t/8,o=(r=function(n){return n>48},i(function(n,t){return n||(r(t)?t:n)},null,n));if(o)throw new Error(o+" bit range requested exceeds 48 bit (6 byte) Number max.");return new e(function(t,r){var e=u+r;return e>t.length?x(r,u.toString()+" bytes"):b(e,i(function(n,t){var r=f(t,n.buf);return {coll:n.coll.concat(r.v),buf:r.buf}},{coll:[],buf:t.slice(r,e)},n).coll)})}function h(n,t){return new e(function(r,e){return s(),e+t>r.length?x(e,t+" bytes for "+n):b(e+t,r.slice(e,e+t))})}function p(n,t){if("number"!=typeof(r=t)||Math.floor(r)!==r||t<0||t>6)throw new Error(n+" requires integer length in range [0, 6].");var r;}function d(n){return p("uintBE",n),h("uintBE("+n+")",n).map(function(t){return t.readUIntBE(0,n)})}function v(n){return p("uintLE",n),h("uintLE("+n+")",n).map(function(t){return t.readUIntLE(0,n)})}function g(n){return p("intBE",n),h("intBE("+n+")",n).map(function(t){return t.readIntBE(0,n)})}function m(n){return p("intLE",n),h("intLE("+n+")",n).map(function(t){return t.readIntLE(0,n)})}function y(n){return n instanceof e}function E(n){return "[object Array]"==={}.toString.call(n)}function w(n){return c()&&Buffer.isBuffer(n)}function b(n,t){return {status:!0,index:n,value:t,furthest:-1,expected:[]}}function x(n,t){return E(t)||(t=[t]),{status:!1,index:-1,value:null,furthest:n,expected:t}}function B(n,t){if(!t)return n;if(n.furthest>t.furthest)return n;var r=n.furthest===t.furthest?function(n,t){if(function(){if(void 0!==e._supportsSet)return e._supportsSet;var n="undefined"!=typeof Set;return e._supportsSet=n,n}()&&Array.from){for(var r=new Set(n),u=0;u=0;){if(i in r){e=r[i].line,0===o&&(o=r[i].lineStart);break}("\n"===n.charAt(i)||"\r"===n.charAt(i)&&"\n"!==n.charAt(i+1))&&(u++,0===o&&(o=i+1)),i--;}var a=e+u,f=t-o;return r[t]={line:a,lineStart:o},{offset:t,line:a+1,column:f+1}}function _(n){if(!y(n))throw new Error("not a parser: "+n)}function L(n,t){return "string"==typeof n?n.charAt(t):n[t]}function O(n){if("number"!=typeof n)throw new Error("not a number: "+n)}function k(n){if("function"!=typeof n)throw new Error("not a function: "+n)}function P(n){if("string"!=typeof n)throw new Error("not a string: "+n)}var q=2,A=3,I=8,F=5*I,M=4*I,z=" ";function R(n,t){return new Array(t+1).join(n)}function U(n,t,r){var e=t-n.length;return e<=0?n:R(r,e)+n}function W(n,t,r,e){return {from:n-t>0?n-t:0,to:n+r>e?e:n+r}}function D(n,t){var r,e,u,o,f,c=t.index,s=c.offset,l=1;if(s===n.length)return "Got the end of the input";if(w(n)){var h=s-s%I,p=s-h,d=W(h,F,M+I,n.length),v=a(function(n){return a(function(n){return U(n.toString(16),2,"0")},n)},function(n,t){var r=n.length,e=[],u=0;if(r<=t)return [n.slice()];for(var o=0;o=4&&(r+=1),l=2,u=a(function(n){return n.length<=4?n.join(" "):n.slice(0,4).join(" ")+" "+n.slice(4).join(" ")},v),(f=(8*(o.to>0?o.to-1:o.to)).toString(16).length)<2&&(f=2);}else {var g=n.split(/\r\n|[\n\r\u2028\u2029]/);r=c.column-1,e=c.line-1,o=W(e,q,A,g.length),u=g.slice(o.from,o.to),f=o.to.toString().length;}var m=e-o.from;return w(n)&&(f=(8*(o.to>0?o.to-1:o.to)).toString(16).length)<2&&(f=2),i(function(t,e,u){var i,a=u===m,c=a?"> ":z;return i=w(n)?U((8*(o.from+u)).toString(16),f,"0"):U((o.from+u+1).toString(),f," "),[].concat(t,[c+i+" | "+e],a?[z+R(" ",f)+" | "+U("",r," ")+R("^",l)]:[])},[],u).join("\n")}function N(n,t){return ["\n","-- PARSING FAILED "+R("-",50),"\n\n",D(n,t),"\n\n",(r=t.expected,1===r.length?"Expected:\n\n"+r[0]:"Expected one of the following: \n\n"+r.join(", ")),"\n"].join("");var r;}function G(n){return void 0!==n.flags?n.flags:[n.global?"g":"",n.ignoreCase?"i":"",n.multiline?"m":"",n.unicode?"u":"",n.sticky?"y":""].join("")}function C(){for(var n=[].slice.call(arguments),t=n.length,r=0;r=2?O(t):t=0;var r=function(n){return RegExp("^(?:"+n.source+")",G(n))}(n),u=""+n;return e(function(n,e){var o=r.exec(n.slice(e));if(o){if(0<=t&&t<=o.length){var i=o[0],a=o[t];return b(e+i.length,a)}return x(e,"valid match group (0 to "+o.length+") in "+u)}return x(e,u)})}function X(n){return e(function(t,r){return b(r,n)})}function Y(n){return e(function(t,r){return x(r,n)})}function Z(n){if(y(n))return e(function(t,r){var e=n._(t,r);return e.index=r,e.value="",e});if("string"==typeof n)return Z(K(n));if(n instanceof RegExp)return Z(Q(n));throw new Error("not a string, regexp, or parser: "+n)}function $(n){return _(n),e(function(t,r){var e=n._(t,r),u=t.slice(r,e.index);return e.status?x(r,'not "'+u+'"'):b(r,null)})}function nn(n){return k(n),e(function(t,r){var e=L(t,r);return r=n.length?x(t,"any character/byte"):b(t+1,L(n,t))}),on=e(function(n,t){return b(n.length,n.slice(t))}),an=e(function(n,t){return t=0}).desc(t)},e.optWhitespace=hn,e.Parser=e,e.range=function(n,t){return nn(function(r){return n<=r&&r<=t}).desc(n+"-"+t)},e.regex=Q,e.regexp=Q,e.sepBy=V,e.sepBy1=H,e.seq=C,e.seqMap=J,e.seqObj=function(){for(var n,t={},r=0,u=(n=arguments,Array.prototype.slice.call(n)),o=u.length,i=0;i255)throw new Error("Value specified to byte constructor ("+n+"=0x"+n.toString(16)+") is larger in value than a single byte.");var t=(n>15?"0x":"0x0")+n.toString(16);return e(function(r,e){var u=L(r,e);return u===n?b(e+1,u):x(e,t)})},buffer:function(n){return h("buffer",n).map(function(n){return Buffer.from(n)})},encodedString:function(n,t){return h("string",t).map(function(t){return t.toString(n)})},uintBE:d,uint8BE:d(1),uint16BE:d(2),uint32BE:d(4),uintLE:v,uint8LE:v(1),uint16LE:v(2),uint32LE:v(4),intBE:g,int8BE:g(1),int16BE:g(2),int32BE:g(4),intLE:m,int8LE:m(1),int16LE:m(2),int32LE:m(4),floatBE:h("floatBE",4).map(function(n){return n.readFloatBE(0)}),floatLE:h("floatLE",4).map(function(n){return n.readFloatLE(0)}),doubleBE:h("doubleBE",8).map(function(n){return n.readDoubleBE(0)}),doubleLE:h("doubleLE",8).map(function(n){return n.readDoubleLE(0)})},n.exports=e;}])}); +}(parsimmon_umd_min)); + +var Parsimmon = /*@__PURE__*/getDefaultExportFromCjs(parsimmon_umd_min.exports); + +class UnknownKeysEntity extends IEntity { + + static attributes = { + lookbehind: + { + value: "", + showDefault: false, + ignore: true, + }, + } + + static { + this.cleanupAttributes(this.attributes); + } + + constructor(values) { + super(values); + /** @type {String} */ this.lookbehind; + } +} + // @ts-nocheck /** @@ -3153,33 +3188,6 @@ class Grammar { /** @param {Grammar} r */ static getGrammarForType(r, attribute, defaultGrammar = r.AttributeAnyValue) { - if (attribute.constructor === Object) { - attribute = /** @type {AttributeInformation} */(attribute); - let type = attribute.type; - let result; - if (type instanceof Array) { - result = Grammar.getGrammarForType(r, type[0]) - .trim(P.optWhitespace) - .sepBy(P.string(",")) - .skip(P.regex(/,?\s*/)) - .wrap(P.string("("), P.string(")")); - } else if (type instanceof UnionType) { - result = type.types - .map(v => Grammar.getGrammarForType(r, Utility.getType(v))) - .reduce((accum, cur) => !cur || accum === r.AttributeAnyValue - ? r.AttributeAnyValue - : accum.or(cur)); - } else { - result = Grammar.getGrammarForType(r, type, defaultGrammar); - } - if (attribute.serialized && !(type instanceof String)) { - result = result.wrap(P.string('"'), P.string('"')); - } - if (attribute.nullable) { - result = result.or(r.Null); - } - return result - } switch (attribute) { case BigInt: return r.BigInt @@ -3189,6 +3197,8 @@ class Grammar { return r.Byte case EnumEntity: return r.Enum + case FormatTextEntity: + return r.FormatText case FunctionReferenceEntity: return r.FunctionReference case GuidEntity: @@ -3241,9 +3251,35 @@ class Grammar { return r.Vector2D case VectorEntity: return r.Vector - default: - return defaultGrammar } + let result = defaultGrammar; + const type = attribute.constructor === Object + ? attribute.type + : attribute.constructor; + if (type instanceof Array) { + result = Grammar.getGrammarForType(r, type[0]) + .trim(P.optWhitespace) + .sepBy(P.string(",")) + .skip(P.regex(/,?\s*/)) + .wrap(P.string("("), P.string(")")); + } else if (type instanceof UnionType) { + result = type.types + .map(v => Grammar.getGrammarForType(r, v)) + .reduce((accum, cur) => !cur || accum === r.AttributeAnyValue + ? r.AttributeAnyValue + : accum.or(cur)); + } else if (attribute.constructor === Object) { + result = Grammar.getGrammarForType(r, type, defaultGrammar); + } + if (attribute.constructor === Object) { + if (attribute.serialized && type.constructor !== String) { + result = result.wrap(P.string('"'), P.string('"')); + } + if (attribute.nullable) { + result = result.or(r.Null); + } + } + return result } /** @param {Grammar} r */ @@ -3286,7 +3322,7 @@ class Grammar { /** * @param {Grammar} r * @param {EntityConstructor} entityType - * @param {Boolean | Number} acceptUnknownKeys can be anumber to specify the limit or true, to let it be a reasonable value + * @param {Boolean | Number} acceptUnknownKeys Number to specify the limit or true, to let it be a reasonable value */ static createEntityGrammar = (r, entityType, acceptUnknownKeys = true) => P.seqMap( @@ -3430,15 +3466,19 @@ class Grammar { namespace: matchResult[1], key: matchResult[2], value: matchResult[3] - } - )) + }) + ) /** @param {Grammar} r */ - InvariantText = r => r.String.trim(P.optWhitespace).wrap( - P.string(InvariantTextEntity.lookbehind).skip(P.optWhitespace).skip(P.string("(")), - P.string(")") + InvariantText = r => + Grammar.regexMap( + new RegExp(String.raw`${InvariantTextEntity.lookbehind}\s*\("(${Grammar.Regex.InsideString.source})"\)`), + matchResult => new InvariantTextEntity({ value: matchResult[1] }) + ) + + FormatText = r => P.string(FormatTextEntity.lookbehind).then(P.optWhitespace).then( + Grammar.getGrammarForType(r, FormatTextEntity.attributes.value).wrap(P.string("("), P.string(")")) ) - .map(value => new InvariantTextEntity({ value: value })) /** @param {Grammar} r */ AttributeAnyValue = r => P.alt( @@ -3667,146 +3707,146 @@ class Grammar { return result } ) -} - -/** - * @typedef {import("../entity/IEntity").EntityConstructor} EntityConstructor - * @typedef {import("../entity/IEntity").AnyValue} AnyValue - * @typedef {import("../entity/IEntity").AnyValueConstructor<*>} AnyValueConstructor - */ - -/** @template {AnyValue} T */ -class ISerializer { - - static grammar = Parsimmon.createLanguage(new Grammar()) - - /** @param {AnyValueConstructor} entityType */ - constructor( - entityType, - attributePrefix = "", - attributeSeparator = ",", - trailingSeparator = false, - attributeValueConjunctionSign = "=", - attributeKeyPrinter = k => k.join(".") - ) { - this.entityType = entityType; - this.attributePrefix = attributePrefix; - this.attributeSeparator = attributeSeparator; - this.trailingSeparator = trailingSeparator; - this.attributeValueConjunctionSign = attributeValueConjunctionSign; - this.attributeKeyPrinter = attributeKeyPrinter; - } - - /** - * @param {String} value - * @returns {T} - */ - deserialize(value) { - return this.read(value) - } - - /** @param {T} object */ - serialize(object, insideString = false, entity = object) { - return this.write(entity, object, insideString) - } - - /** - * @protected - * @param {String} value - * @returns {T} - */ - read(value) { - throw new Error("Not implemented") - } - - /** - * @protected - * @param {T} object - * @param {Boolean} insideString - * @returns {String} - */ - write(entity, object, insideString) { - throw new Error("Not implemented") - } - - /** - * @protected - * @param {AnyValue} value - * @param {String[]} fullKey - * @param {Boolean} insideString - */ - writeValue(entity, value, fullKey, insideString) { - const type = Utility.getType(value); - // @ts-expect-error - const serializer = SerializerFactory.getSerializer(type); - if (!serializer) { - throw new Error(`Unknown value type "${type.name}", a serializer must be registered in the SerializerFactory class, check initializeSerializerFactory.js`) - } - return serializer.write( - value instanceof IEntity ? value : entity, - value, - insideString - ) - } - - /** - * @protected - * @param {String[]} key - * @param {Object} object - * @param {Boolean} insideString - * @returns {String} - */ - subWrite(entity, key, object, insideString) { - let result = ""; - let fullKey = key.concat(""); - const last = fullKey.length - 1; - const attributes = /** @type {EntityConstructor} */(object.constructor).attributes; - const keys = attributes - ? Utility.mergeArrays( - Object.keys(attributes), - Object.keys(object) - ) - : Object.keys(object); - for (const property of keys) { - fullKey[last] = property; - const value = object[property]; - if (value?.constructor === Object) { - // Recursive call when finding an object - result += (result.length ? this.attributeSeparator : "") - + this.subWrite(entity, fullKey, value, insideString); - } else if (value !== undefined && this.showProperty(entity, object, fullKey, value)) { - const isSerialized = Utility.isSerialized(entity, fullKey); - result += (result.length ? this.attributeSeparator : "") - + this.attributePrefix - + this.attributeKeyPrinter(fullKey) - + this.attributeValueConjunctionSign - + ( - isSerialized - ? `"${this.writeValue(entity, value, fullKey, true)}"` - : this.writeValue(entity, value, fullKey, insideString) - ); - } - } - if (this.trailingSeparator && result.length && fullKey.length === 1) { - // append separator at the end if asked and there was printed content - result += this.attributeSeparator; - } - return result - } - - showProperty(entity, object, attributeKey, attributeValue) { - const attributes = /** @type {EntityConstructor} */(this.entityType).attributes; - const attribute = Utility.objectGet(attributes, attributeKey); - if (attribute?.constructor === Object) { - if (attribute.ignored) { - return false - } - return !Utility.equals(attribute.value, attributeValue) || attribute.showDefault - } - return true - } -} - +} + +/** + * @typedef {import("../entity/IEntity").EntityConstructor} EntityConstructor + * @typedef {import("../entity/IEntity").AnyValue} AnyValue + * @typedef {import("../entity/IEntity").AnyValueConstructor<*>} AnyValueConstructor + */ + +/** @template {AnyValue} T */ +class ISerializer { + + static grammar = Parsimmon.createLanguage(new Grammar()) + + /** @param {AnyValueConstructor} entityType */ + constructor( + entityType, + attributePrefix = "", + attributeSeparator = ",", + trailingSeparator = false, + attributeValueConjunctionSign = "=", + attributeKeyPrinter = k => k.join(".") + ) { + this.entityType = entityType; + this.attributePrefix = attributePrefix; + this.attributeSeparator = attributeSeparator; + this.trailingSeparator = trailingSeparator; + this.attributeValueConjunctionSign = attributeValueConjunctionSign; + this.attributeKeyPrinter = attributeKeyPrinter; + } + + /** + * @param {String} value + * @returns {T} + */ + deserialize(value) { + return this.read(value) + } + + /** @param {T} object */ + serialize(object, insideString = false, entity = object) { + return this.write(entity, object, insideString) + } + + /** + * @protected + * @param {String} value + * @returns {T} + */ + read(value) { + throw new Error("Not implemented") + } + + /** + * @protected + * @param {T} object + * @param {Boolean} insideString + * @returns {String} + */ + write(entity, object, insideString) { + throw new Error("Not implemented") + } + + /** + * @protected + * @param {AnyValue} value + * @param {String[]} fullKey + * @param {Boolean} insideString + */ + writeValue(entity, value, fullKey, insideString) { + const type = Utility.getType(value); + // @ts-expect-error + const serializer = SerializerFactory.getSerializer(type); + if (!serializer) { + throw new Error(`Unknown value type "${type.name}", a serializer must be registered in the SerializerFactory class, check initializeSerializerFactory.js`) + } + return serializer.write( + value instanceof IEntity ? value : entity, + value, + insideString + ) + } + + /** + * @protected + * @param {String[]} key + * @param {Object} object + * @param {Boolean} insideString + * @returns {String} + */ + subWrite(entity, key, object, insideString) { + let result = ""; + let fullKey = key.concat(""); + const last = fullKey.length - 1; + const attributes = /** @type {EntityConstructor} */(object.constructor).attributes; + const keys = attributes + ? Utility.mergeArrays( + Object.keys(attributes), + Object.keys(object) + ) + : Object.keys(object); + for (const property of keys) { + fullKey[last] = property; + const value = object[property]; + if (value?.constructor === Object) { + // Recursive call when finding an object + result += (result.length ? this.attributeSeparator : "") + + this.subWrite(entity, fullKey, value, insideString); + } else if (value !== undefined && this.showProperty(entity, object, fullKey, value)) { + const isSerialized = Utility.isSerialized(entity, fullKey); + result += (result.length ? this.attributeSeparator : "") + + this.attributePrefix + + this.attributeKeyPrinter(fullKey) + + this.attributeValueConjunctionSign + + ( + isSerialized + ? `"${this.writeValue(entity, value, fullKey, true)}"` + : this.writeValue(entity, value, fullKey, insideString) + ); + } + } + if (this.trailingSeparator && result.length && fullKey.length === 1) { + // append separator at the end if asked and there was printed content + result += this.attributeSeparator; + } + return result + } + + showProperty(entity, object, attributeKey, attributeValue) { + const attributes = /** @type {EntityConstructor} */(this.entityType).attributes; + const attribute = Utility.objectGet(attributes, attributeKey); + if (attribute?.constructor === Object) { + if (attribute.ignored) { + return false + } + return !Utility.equals(attribute.value, attributeValue) || attribute.showDefault + } + return true + } +} + class ObjectSerializer extends ISerializer { constructor() { @@ -3863,8 +3903,8 @@ ${this.subWrite(entity, [], object, insideString) End Object\n`; return result } -} - +} + class Copy extends IInput { static #serializer = new ObjectSerializer() @@ -3899,187 +3939,187 @@ class Copy extends IInput { const value = this.getSerializedText(); navigator.clipboard.writeText(value); } -} - -/** - * @typedef {import("../element/IElement").default} IElement - * @typedef {import("../input/IInput").default} IInput - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -/** @template {IElement} T */ -class ITemplate { - - /** @type {T} */ - element - - get blueprint() { - return this.element.blueprint - } - - /** @type {IInput[]} */ - #inputObjects = [] - get inputObjects() { - return this.#inputObjects - } - - /** @param {T} element */ - initialize(element) { - this.element = element; - } - - createInputObjects() { - return /** @type {IInput[]} */([]) - } - - /** - * @template {IInput} T - * @param {new () => T} type - */ - getInputObject(type) { - return /** @type {T} */(this.inputObjects.find(object => object.constructor == type)) - } - - setup() { - this.#inputObjects.forEach(v => v.setup()); - } - - cleanup() { - this.#inputObjects.forEach(v => v.cleanup()); - } - - /** @param {PropertyValues} changedProperties */ - willUpdate(changedProperties) { - } - - /** @param {PropertyValues} changedProperties */ - update(changedProperties) { - } - - render() { - return y`` - } - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - } - - /** @param {PropertyValues} changedProperties */ - updated(changedProperties) { - } - - inputSetup() { - this.#inputObjects = this.createInputObjects(); - } -} - -/** @typedef {import("../../Blueprint").default} Blueprint */ - -/** - * @template {HTMLElement} T - * @extends IInput - */ -class IKeyboardShortcut extends IInput { - - /** @type {KeyBindingEntity[]} */ - #activationKeys - - /** - * @param {T} target - * @param {Blueprint} blueprint - * @param {Object} options - */ - constructor(target, blueprint, options = {}) { - options.activateAnyKey ??= false; - options.activationKeys ??= []; - options.consumeEvent ??= true; - options.listenOnFocus ??= true; - options.unlistenOnTextEdit ??= true; // No shortcuts when inside of a text field - if (!(options.activationKeys instanceof Array)) { - options.activationKeys = [options.activationKeys]; - } - options.activationKeys = options.activationKeys.map(v => { - if (v instanceof KeyBindingEntity) { - return v - } - if (v.constructor === String) { - const parsed = ISerializer.grammar.KeyBinding.parse(v); - if (parsed.status) { - return parsed.value - } - } - throw new Error("Unexpected key value") - }); - - super(target, blueprint, options); - - this.#activationKeys = this.options.activationKeys ?? []; - - const wantsShift = keyEntry => keyEntry.bShift || keyEntry.Key == "LeftShift" || keyEntry.Key == "RightShift"; - const wantsCtrl = keyEntry => keyEntry.bCtrl || keyEntry.Key == "LeftControl" || keyEntry.Key == "RightControl"; - const wantsAlt = keyEntry => keyEntry.bAlt || keyEntry.Key == "LeftAlt" || keyEntry.Key == "RightAlt"; - - let self = this; - /** @param {KeyboardEvent} e */ - this.keyDownHandler = e => { - if ( - this.options.activateAnyKey - || self.#activationKeys.some(keyEntry => - wantsShift(keyEntry) == e.shiftKey - && wantsCtrl(keyEntry) == e.ctrlKey - && wantsAlt(keyEntry) == e.altKey - && Configuration.Keys[keyEntry.Key] == e.code - ) - ) { - if (options.consumeEvent) { - e.preventDefault(); - e.stopImmediatePropagation(); - } - self.fire(); - document.removeEventListener("keydown", self.keyDownHandler); - document.addEventListener("keyup", self.keyUpHandler); - } - }; - - /** @param {KeyboardEvent} e */ - this.keyUpHandler = e => { - if ( - this.options.activateAnyKey - || self.#activationKeys.some(keyEntry => - keyEntry.bShift && e.key == "Shift" - || keyEntry.bCtrl && e.key == "Control" - || keyEntry.bAlt && e.key == "Alt" - || keyEntry.bCmd && e.key == "Meta" - || Configuration.Keys[keyEntry.Key] == e.code - ) - ) { - if (options.consumeEvent) { - e.stopImmediatePropagation(); - } - self.unfire(); - document.removeEventListener("keyup", this.keyUpHandler); - document.addEventListener("keydown", this.keyDownHandler); - } - }; - } - - listenEvents() { - document.addEventListener("keydown", this.keyDownHandler); - } - - unlistenEvents() { - document.removeEventListener("keydown", this.keyDownHandler); - } - - // Subclasses will want to override - - fire() { - } - - unfire() { - } -} - +} + +/** + * @typedef {import("../element/IElement").default} IElement + * @typedef {import("../input/IInput").default} IInput + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +/** @template {IElement} T */ +class ITemplate { + + /** @type {T} */ + element + + get blueprint() { + return this.element.blueprint + } + + /** @type {IInput[]} */ + #inputObjects = [] + get inputObjects() { + return this.#inputObjects + } + + /** @param {T} element */ + initialize(element) { + this.element = element; + } + + createInputObjects() { + return /** @type {IInput[]} */([]) + } + + /** + * @template {IInput} T + * @param {new () => T} type + */ + getInputObject(type) { + return /** @type {T} */(this.inputObjects.find(object => object.constructor == type)) + } + + setup() { + this.#inputObjects.forEach(v => v.setup()); + } + + cleanup() { + this.#inputObjects.forEach(v => v.cleanup()); + } + + /** @param {PropertyValues} changedProperties */ + willUpdate(changedProperties) { + } + + /** @param {PropertyValues} changedProperties */ + update(changedProperties) { + } + + render() { + return y`` + } + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + } + + /** @param {PropertyValues} changedProperties */ + updated(changedProperties) { + } + + inputSetup() { + this.#inputObjects = this.createInputObjects(); + } +} + +/** @typedef {import("../../Blueprint").default} Blueprint */ + +/** + * @template {HTMLElement} T + * @extends IInput + */ +class IKeyboardShortcut extends IInput { + + /** @type {KeyBindingEntity[]} */ + #activationKeys + + /** + * @param {T} target + * @param {Blueprint} blueprint + * @param {Object} options + */ + constructor(target, blueprint, options = {}) { + options.activateAnyKey ??= false; + options.activationKeys ??= []; + options.consumeEvent ??= true; + options.listenOnFocus ??= true; + options.unlistenOnTextEdit ??= true; // No shortcuts when inside of a text field + if (!(options.activationKeys instanceof Array)) { + options.activationKeys = [options.activationKeys]; + } + options.activationKeys = options.activationKeys.map(v => { + if (v instanceof KeyBindingEntity) { + return v + } + if (v.constructor === String) { + const parsed = ISerializer.grammar.KeyBinding.parse(v); + if (parsed.status) { + return parsed.value + } + } + throw new Error("Unexpected key value") + }); + + super(target, blueprint, options); + + this.#activationKeys = this.options.activationKeys ?? []; + + const wantsShift = keyEntry => keyEntry.bShift || keyEntry.Key == "LeftShift" || keyEntry.Key == "RightShift"; + const wantsCtrl = keyEntry => keyEntry.bCtrl || keyEntry.Key == "LeftControl" || keyEntry.Key == "RightControl"; + const wantsAlt = keyEntry => keyEntry.bAlt || keyEntry.Key == "LeftAlt" || keyEntry.Key == "RightAlt"; + + let self = this; + /** @param {KeyboardEvent} e */ + this.keyDownHandler = e => { + if ( + this.options.activateAnyKey + || self.#activationKeys.some(keyEntry => + wantsShift(keyEntry) == e.shiftKey + && wantsCtrl(keyEntry) == e.ctrlKey + && wantsAlt(keyEntry) == e.altKey + && Configuration.Keys[keyEntry.Key] == e.code + ) + ) { + if (options.consumeEvent) { + e.preventDefault(); + e.stopImmediatePropagation(); + } + self.fire(); + document.removeEventListener("keydown", self.keyDownHandler); + document.addEventListener("keyup", self.keyUpHandler); + } + }; + + /** @param {KeyboardEvent} e */ + this.keyUpHandler = e => { + if ( + this.options.activateAnyKey + || self.#activationKeys.some(keyEntry => + keyEntry.bShift && e.key == "Shift" + || keyEntry.bCtrl && e.key == "Control" + || keyEntry.bAlt && e.key == "Alt" + || keyEntry.bCmd && e.key == "Meta" + || Configuration.Keys[keyEntry.Key] == e.code + ) + ) { + if (options.consumeEvent) { + e.stopImmediatePropagation(); + } + self.unfire(); + document.removeEventListener("keyup", this.keyUpHandler); + document.addEventListener("keydown", this.keyDownHandler); + } + }; + } + + listenEvents() { + document.addEventListener("keydown", this.keyDownHandler); + } + + unlistenEvents() { + document.removeEventListener("keydown", this.keyDownHandler); + } + + // Subclasses will want to override + + fire() { + } + + unfire() { + } +} + class KeyboardCanc extends IKeyboardShortcut { /** @@ -4095,75 +4135,75 @@ class KeyboardCanc extends IKeyboardShortcut { fire() { this.blueprint.removeGraphElement(...this.blueprint.getNodes(true)); } -} - -/** - * @template {HTMLElement} T - * @extends {IInput} - */ -class IPointing extends IInput { - - constructor(target, blueprint, options = {}) { - options.ignoreTranslateCompensate ??= false; - options.ignoreScale ??= false; - options.movementSpace ??= blueprint.getGridDOMElement() ?? document.documentElement; - super(target, blueprint, options); - /** @type {HTMLElement} */ - this.movementSpace = options.movementSpace; - } - - /** @param {MouseEvent} mouseEvent */ - locationFromEvent(mouseEvent) { - const location = Utility.convertLocation( - [mouseEvent.clientX, mouseEvent.clientY], - this.movementSpace, - this.options.ignoreScale - ); - return this.options.ignoreTranslateCompensate - ? location - : this.blueprint.compensateTranslation(location[0], location[1]) - } -} - -class IMouseWheel extends IPointing { - - /** @param {WheelEvent} e */ - #mouseWheelHandler = e => { - e.preventDefault(); - const location = this.locationFromEvent(e); - this.wheel(Math.sign(e.deltaY * Configuration.mouseWheelFactor), location); - } - - /** @param {WheelEvent} e */ - #mouseParentWheelHandler = e => e.preventDefault() - - /** - * @param {HTMLElement} target - * @param {import("../../Blueprint").default} blueprint - * @param {Object} options - */ - constructor(target, blueprint, options = {}) { - options.listenOnFocus = true; - options.strictTarget ??= false; - super(target, blueprint, options); - this.strictTarget = options.strictTarget; - } - - listenEvents() { - this.movementSpace.addEventListener("wheel", this.#mouseWheelHandler, false); - this.movementSpace.parentElement?.addEventListener("wheel", this.#mouseParentWheelHandler); - } - - unlistenEvents() { - this.movementSpace.removeEventListener("wheel", this.#mouseWheelHandler, false); - this.movementSpace.parentElement?.removeEventListener("wheel", this.#mouseParentWheelHandler); - } - - /* Subclasses will override the following method */ - wheel(variation, location) { - } -} - +} + +/** + * @template {HTMLElement} T + * @extends {IInput} + */ +class IPointing extends IInput { + + constructor(target, blueprint, options = {}) { + options.ignoreTranslateCompensate ??= false; + options.ignoreScale ??= false; + options.movementSpace ??= blueprint.getGridDOMElement() ?? document.documentElement; + super(target, blueprint, options); + /** @type {HTMLElement} */ + this.movementSpace = options.movementSpace; + } + + /** @param {MouseEvent} mouseEvent */ + locationFromEvent(mouseEvent) { + const location = Utility.convertLocation( + [mouseEvent.clientX, mouseEvent.clientY], + this.movementSpace, + this.options.ignoreScale + ); + return this.options.ignoreTranslateCompensate + ? location + : this.blueprint.compensateTranslation(location[0], location[1]) + } +} + +class IMouseWheel extends IPointing { + + /** @param {WheelEvent} e */ + #mouseWheelHandler = e => { + e.preventDefault(); + const location = this.locationFromEvent(e); + this.wheel(Math.sign(e.deltaY * Configuration.mouseWheelFactor), location); + } + + /** @param {WheelEvent} e */ + #mouseParentWheelHandler = e => e.preventDefault() + + /** + * @param {HTMLElement} target + * @param {import("../../Blueprint").default} blueprint + * @param {Object} options + */ + constructor(target, blueprint, options = {}) { + options.listenOnFocus = true; + options.strictTarget ??= false; + super(target, blueprint, options); + this.strictTarget = options.strictTarget; + } + + listenEvents() { + this.movementSpace.addEventListener("wheel", this.#mouseWheelHandler, false); + this.movementSpace.parentElement?.addEventListener("wheel", this.#mouseParentWheelHandler); + } + + unlistenEvents() { + this.movementSpace.removeEventListener("wheel", this.#mouseWheelHandler, false); + this.movementSpace.parentElement?.removeEventListener("wheel", this.#mouseParentWheelHandler); + } + + /* Subclasses will override the following method */ + wheel(variation, location) { + } +} + class Zoom extends IMouseWheel { #enableZoonIn = false @@ -4186,33 +4226,33 @@ class Zoom extends IMouseWheel { zoomLevel += variation; this.blueprint.setZoom(zoomLevel, location); } -} - -class KeyboardEnableZoom extends IKeyboardShortcut { - - /** @type {Zoom} */ - #zoomInputObject - - /** - * @param {HTMLElement} target - * @param {import("../../Blueprint").default} blueprint - * @param {Object} options - */ - constructor(target, blueprint, options = {}) { - options.activationKeys = Configuration.enableZoomIn; - super(target, blueprint, options); - } - - fire() { - this.#zoomInputObject = this.blueprint.getInputObject(Zoom); - this.#zoomInputObject.enableZoonIn = true; - } - - unfire() { - this.#zoomInputObject.enableZoonIn = false; - } -} - +} + +class KeyboardEnableZoom extends IKeyboardShortcut { + + /** @type {Zoom} */ + #zoomInputObject + + /** + * @param {HTMLElement} target + * @param {import("../../Blueprint").default} blueprint + * @param {Object} options + */ + constructor(target, blueprint, options = {}) { + options.activationKeys = Configuration.enableZoomIn; + super(target, blueprint, options); + } + + fire() { + this.#zoomInputObject = this.blueprint.getInputObject(Zoom); + this.#zoomInputObject.enableZoonIn = true; + } + + unfire() { + this.#zoomInputObject.enableZoonIn = false; + } +} + /** @typedef {import("../../Blueprint").default} Blueprint */ class KeyboardSelectAll extends IKeyboardShortcut { @@ -4229,489 +4269,489 @@ class KeyboardSelectAll extends IKeyboardShortcut { fire() { this.blueprint.selectAll(); } -} - -/** - * @typedef {import("../Blueprint").default} Blueprint - * @typedef {import("../entity/IEntity").default} IEntity - * @typedef {import("../input/IInput").default} IInput - * @typedef {import("../template/ITemplate").default} ITemplate - * @typedef {import("lit").PropertyDeclarations} PropertyDeclarations - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -/** - * @template {IEntity} T - * @template {ITemplate} U - */ -class IElement extends s { - - #nextUpdatedCallbacks = [] - - /** @type {Blueprint} */ - #blueprint - get blueprint() { - return this.#blueprint - } - set blueprint(v) { - this.#blueprint = v; - } - - /** @type {T} */ - #entity - get entity() { - return this.#entity - } - set entity(entity) { - this.#entity = entity; - } - - /** @type {U} */ - #template - get template() { - return this.#template - } - - isInitialized = false - isSetup = false - - /** @type {IInput[]} */ - inputObjects = [] - - /** - * @param {T} entity - * @param {U} template - */ - initialize(entity, template) { - this.requestUpdate(); - this.#entity = entity; - this.#template = template; - this.#template.initialize(this); - if (this.isConnected) { - this.updateComplete.then(() => this.setup()); - } - this.isInitialized = true; - } - - connectedCallback() { - super.connectedCallback(); - this.blueprint = /** @type {Blueprint} */(this.closest("ueb-blueprint")); - if (this.isInitialized) { - this.requestUpdate(); - this.updateComplete.then(() => this.setup()); - } - } - - disconnectedCallback() { - super.disconnectedCallback(); - if (this.isSetup) { - this.updateComplete.then(() => this.cleanup()); - } - this.acknowledgeDelete(); - } - - createRenderRoot() { - return this - } - - /** @param {PropertyValues} changedProperties */ - shouldUpdate(changedProperties) { - return this.isInitialized && this.isConnected - } - - setup() { - this.template.setup(); - this.isSetup = true; - } - - cleanup() { - this.template.cleanup(); - this.isSetup = false; - } - - /** @param {PropertyValues} changedProperties */ - willUpdate(changedProperties) { - super.willUpdate(changedProperties); - this.template.willUpdate(changedProperties); - } - - /** @param {PropertyValues} changedProperties */ - update(changedProperties) { - super.update(changedProperties); - this.template.update(changedProperties); - } - - render() { - return this.template.render() - } - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - this.template.firstUpdated(changedProperties); - this.template.inputSetup(); - } - - /** @param {PropertyValues} changedProperties */ - updated(changedProperties) { - super.updated(changedProperties); - this.template.updated(changedProperties); - // Remember the array might change while iterating - for (const f of this.#nextUpdatedCallbacks) { - f(changedProperties); - } - this.#nextUpdatedCallbacks = []; - } - - addNextUpdatedCallbacks(callback, requestUpdate = false) { - this.#nextUpdatedCallbacks.push(callback); - if (requestUpdate) { - this.requestUpdate(); - } - } - - acknowledgeDelete() { - let deleteEvent = new CustomEvent(Configuration.removeEventName); - this.dispatchEvent(deleteEvent); - } - - /** @param {IElement} element */ - isSameGraph(element) { - return this.blueprint && this.blueprint == element?.blueprint - } - - /** - * @template {IInput} V - * @param {new (...args: any[]) => V} type - */ - getInputObject(type) { - return /** @type {V} */(this.template.inputObjects.find(object => object.constructor == type)) - } -} - -/** - * @typedef {import("../entity/IEntity").default} IEntity - * @typedef {import("../template/IDraggableTemplate").default} IDraggableTemplate - * @typedef {CustomEvent<{ - * value: [Number, Number] - * }>} DragEvent - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -/** - * @template {IEntity} T - * @template {IDraggableTemplate} U - * @extends {IElement} - */ -class IDraggableElement extends IElement { - - static properties = { - ...super.properties, - locationX: { - type: Number, - attribute: false, - }, - locationY: { - type: Number, - attribute: false, - }, - sizeX: { - type: Number, - attribute: false, - }, - sizeY: { - type: Number, - attribute: false, - }, - } - static dragEventName = Configuration.dragEventName - static dragGeneralEventName = Configuration.dragGeneralEventName - - constructor() { - super(); - this.locationX = 0; - this.locationY = 0; - this.sizeX = 0; - this.sizeY = 0; - } - - computeSizes() { - const bounding = this.getBoundingClientRect(); - this.sizeX = this.blueprint.scaleCorrect(bounding.width); - this.sizeY = this.blueprint.scaleCorrect(bounding.height); - } - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - this.computeSizes(); - } - - /** - * @param {Number} x - * @param {Number} y - */ - setLocation(x, y, acknowledge = true) { - const dx = x - this.locationX; - const dy = y - this.locationY; - this.locationX = x; - this.locationY = y; - if (this.blueprint && acknowledge) { - const dragLocalEvent = new CustomEvent( - /** @type {typeof IDraggableElement} */(this.constructor).dragEventName, - { - detail: { - value: [dx, dy], - }, - bubbles: false, - cancelable: true, - } - ); - this.dispatchEvent(dragLocalEvent); - } - } - - /** - * @param {Number} x - * @param {Number} y - */ - addLocation(x, y, acknowledge = true) { - this.setLocation(this.locationX + x, this.locationY + y, acknowledge); - } - - /** @param {Number[]} value */ - acknowledgeDrag(value) { - const dragEvent = new CustomEvent( - /** @type {typeof IDraggableElement} */(this.constructor).dragGeneralEventName, - { - detail: { - value: value - }, - bubbles: true, - cancelable: true - } - ); - this.dispatchEvent(dragEvent); - } - - snapToGrid() { - const snappedLocation = Utility.snapToGrid(this.locationX, this.locationY, Configuration.gridSize); - if (this.locationX != snappedLocation[0] || this.locationY != snappedLocation[1]) { - this.setLocation(snappedLocation[0], snappedLocation[1]); - } - } - - topBoundary(justSelectableArea = false) { - return this.template.topBoundary(justSelectableArea) - } - - rightBoundary(justSelectableArea = false) { - return this.template.rightBoundary(justSelectableArea) - } - - bottomBoundary(justSelectableArea = false) { - return this.template.bottomBoundary(justSelectableArea) - } - - leftBoundary(justSelectableArea = false) { - return this.template.leftBoundary(justSelectableArea) - } -} - -/** - * @typedef {import("../../Blueprint").default} Blueprint - * @typedef {import("../../element/IElement").default} IElement - */ - -/** - * @template {IElement} T - * @extends {IPointing} - */ -class IMouseClickDrag extends IPointing { - - /** @param {MouseEvent} e */ - #mouseDownHandler = e => { - this.blueprint.setFocused(true); - switch (e.button) { - case this.options.clickButton: - // Either doesn't matter or consider the click only when clicking on the parent, not descandants - if (!this.options.strictTarget || e.target == e.currentTarget) { - if (this.options.consumeEvent) { - e.stopImmediatePropagation(); // Captured, don't call anyone else - } - // Attach the listeners - this.#movementListenedElement.addEventListener("mousemove", this.#mouseStartedMovingHandler); - document.addEventListener("mouseup", this.#mouseUpHandler); - this.clickedPosition = this.locationFromEvent(e); - this.blueprint.mousePosition[0] = this.clickedPosition[0]; - this.blueprint.mousePosition[1] = this.clickedPosition[1]; - if (this.target instanceof IDraggableElement) { - this.clickedOffset = [ - this.clickedPosition[0] - this.target.locationX, - this.clickedPosition[1] - this.target.locationY, - ]; - } - this.clicked(this.clickedPosition); - } - break - default: - if (!this.options.exitAnyButton) { - this.#mouseUpHandler(e); - } - break - } - } - - /** @param {MouseEvent} e */ - #mouseStartedMovingHandler = e => { - if (this.options.consumeEvent) { - e.stopImmediatePropagation(); // Captured, don't call anyone else - } - // Delegate from now on to this.#mouseMoveHandler - this.#movementListenedElement.removeEventListener("mousemove", this.#mouseStartedMovingHandler); - this.#movementListenedElement.addEventListener("mousemove", this.#mouseMoveHandler); - // Handler calls e.preventDefault() when it receives the event, this means dispatchEvent returns false - const dragEvent = this.getEvent(Configuration.trackingMouseEventName.begin); - this.#trackingMouse = this.target.dispatchEvent(dragEvent) == false; - const location = this.locationFromEvent(e); - // Do actual actions - this.lastLocation = Utility.snapToGrid(this.clickedPosition[0], this.clickedPosition[1], this.stepSize); - this.startDrag(location); - this.started = true; - } - - /** @param {MouseEvent} e */ - #mouseMoveHandler = e => { - if (this.options.consumeEvent) { - e.stopImmediatePropagation(); // Captured, don't call anyone else - } - const location = this.locationFromEvent(e); - const movement = [e.movementX, e.movementY]; - this.dragTo(location, movement); - if (this.#trackingMouse) { - this.blueprint.mousePosition = location; - } - if (this.options.scrollGraphEdge) { - const movementNorm = Math.sqrt(movement[0] * movement[0] + movement[1] * movement[1]); - const threshold = this.blueprint.scaleCorrect(Configuration.edgeScrollThreshold); - const leftThreshold = this.blueprint.template.gridLeftVisibilityBoundary() + threshold; - const rightThreshold = this.blueprint.template.gridRightVisibilityBoundary() - threshold; - let scrollX = 0; - if (location[0] < leftThreshold) { - scrollX = location[0] - leftThreshold; - } else if (location[0] > rightThreshold) { - scrollX = location[0] - rightThreshold; - } - const topThreshold = this.blueprint.template.gridTopVisibilityBoundary() + threshold; - const bottomThreshold = this.blueprint.template.gridBottomVisibilityBoundary() - threshold; - let scrollY = 0; - if (location[1] < topThreshold) { - scrollY = location[1] - topThreshold; - } else if (location[1] > bottomThreshold) { - scrollY = location[1] - bottomThreshold; - } - scrollX = Utility.clamp(this.blueprint.scaleCorrectReverse(scrollX) ** 3 * movementNorm * 0.6, -20, 20); - scrollY = Utility.clamp(this.blueprint.scaleCorrectReverse(scrollY) ** 3 * movementNorm * 0.6, -20, 20); - this.blueprint.scrollDelta(scrollX, scrollY); - } - } - - /** @param {MouseEvent} e */ - #mouseUpHandler = e => { - if (!this.options.exitAnyButton || e.button == this.options.clickButton) { - if (this.options.consumeEvent) { - e.stopImmediatePropagation(); // Captured, don't call anyone else - } - // Remove the handlers of "mousemove" and "mouseup" - this.#movementListenedElement.removeEventListener("mousemove", this.#mouseStartedMovingHandler); - this.#movementListenedElement.removeEventListener("mousemove", this.#mouseMoveHandler); - document.removeEventListener("mouseup", this.#mouseUpHandler); - if (this.started) { - this.endDrag(); - } - this.unclicked(); - if (this.#trackingMouse) { - const dragEvent = this.getEvent(Configuration.trackingMouseEventName.end); - this.target.dispatchEvent(dragEvent); - this.#trackingMouse = false; - } - this.started = false; - } - } - - #trackingMouse = false - #movementListenedElement - #draggableElement - - clickedOffset = [0, 0] - clickedPosition = [0, 0] - lastLocation = [0, 0] - started = false - stepSize = 1 - - /** - * @param {T} target - * @param {Blueprint} blueprint - * @param {Object} options - */ - constructor(target, blueprint, options = {}) { - options.clickButton ??= 0; - options.consumeEvent ??= true; - options.draggableElement ??= target; - options.exitAnyButton ??= true; - options.moveEverywhere ??= false; - options.movementSpace ??= blueprint?.getGridDOMElement(); - options.repositionOnClick ??= false; - options.scrollGraphEdge ??= false; - options.strictTarget ??= false; - super(target, blueprint, options); - this.stepSize = parseInt(options?.stepSize ?? Configuration.gridSize); - this.#movementListenedElement = this.options.moveEverywhere ? document.documentElement : this.movementSpace; - this.#draggableElement = /** @type {HTMLElement} */(this.options.draggableElement); - - this.listenEvents(); - } - - listenEvents() { - super.listenEvents(); - this.#draggableElement.addEventListener("mousedown", this.#mouseDownHandler); - if (this.options.clickButton == 2) { - this.#draggableElement.addEventListener("contextmenu", e => e.preventDefault()); - } - } - - unlistenEvents() { - super.unlistenEvents(); - this.#draggableElement.removeEventListener("mousedown", this.#mouseDownHandler); - } - - getEvent(eventName) { - return new CustomEvent(eventName, { - detail: { - tracker: this - }, - bubbles: true, - cancelable: true - }) - } - - /* Subclasses will override the following methods */ - clicked(location) { - } - - startDrag(location) { - } - - dragTo(location, offset) { - } - - endDrag() { - } - - unclicked(location) { - } -} - +} + +/** + * @typedef {import("../Blueprint").default} Blueprint + * @typedef {import("../entity/IEntity").default} IEntity + * @typedef {import("../input/IInput").default} IInput + * @typedef {import("../template/ITemplate").default} ITemplate + * @typedef {import("lit").PropertyDeclarations} PropertyDeclarations + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +/** + * @template {IEntity} T + * @template {ITemplate} U + */ +class IElement extends s { + + #nextUpdatedCallbacks = [] + + /** @type {Blueprint} */ + #blueprint + get blueprint() { + return this.#blueprint + } + set blueprint(v) { + this.#blueprint = v; + } + + /** @type {T} */ + #entity + get entity() { + return this.#entity + } + set entity(entity) { + this.#entity = entity; + } + + /** @type {U} */ + #template + get template() { + return this.#template + } + + isInitialized = false + isSetup = false + + /** @type {IInput[]} */ + inputObjects = [] + + /** + * @param {T} entity + * @param {U} template + */ + initialize(entity, template) { + this.requestUpdate(); + this.#entity = entity; + this.#template = template; + this.#template.initialize(this); + if (this.isConnected) { + this.updateComplete.then(() => this.setup()); + } + this.isInitialized = true; + } + + connectedCallback() { + super.connectedCallback(); + this.blueprint = /** @type {Blueprint} */(this.closest("ueb-blueprint")); + if (this.isInitialized) { + this.requestUpdate(); + this.updateComplete.then(() => this.setup()); + } + } + + disconnectedCallback() { + super.disconnectedCallback(); + if (this.isSetup) { + this.updateComplete.then(() => this.cleanup()); + } + this.acknowledgeDelete(); + } + + createRenderRoot() { + return this + } + + /** @param {PropertyValues} changedProperties */ + shouldUpdate(changedProperties) { + return this.isInitialized && this.isConnected + } + + setup() { + this.template.setup(); + this.isSetup = true; + } + + cleanup() { + this.template.cleanup(); + this.isSetup = false; + } + + /** @param {PropertyValues} changedProperties */ + willUpdate(changedProperties) { + super.willUpdate(changedProperties); + this.template.willUpdate(changedProperties); + } + + /** @param {PropertyValues} changedProperties */ + update(changedProperties) { + super.update(changedProperties); + this.template.update(changedProperties); + } + + render() { + return this.template.render() + } + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + this.template.firstUpdated(changedProperties); + this.template.inputSetup(); + } + + /** @param {PropertyValues} changedProperties */ + updated(changedProperties) { + super.updated(changedProperties); + this.template.updated(changedProperties); + // Remember the array might change while iterating + for (const f of this.#nextUpdatedCallbacks) { + f(changedProperties); + } + this.#nextUpdatedCallbacks = []; + } + + addNextUpdatedCallbacks(callback, requestUpdate = false) { + this.#nextUpdatedCallbacks.push(callback); + if (requestUpdate) { + this.requestUpdate(); + } + } + + acknowledgeDelete() { + let deleteEvent = new CustomEvent(Configuration.removeEventName); + this.dispatchEvent(deleteEvent); + } + + /** @param {IElement} element */ + isSameGraph(element) { + return this.blueprint && this.blueprint == element?.blueprint + } + + /** + * @template {IInput} V + * @param {new (...args: any[]) => V} type + */ + getInputObject(type) { + return /** @type {V} */(this.template.inputObjects.find(object => object.constructor == type)) + } +} + +/** + * @typedef {import("../entity/IEntity").default} IEntity + * @typedef {import("../template/IDraggableTemplate").default} IDraggableTemplate + * @typedef {CustomEvent<{ + * value: [Number, Number] + * }>} DragEvent + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +/** + * @template {IEntity} T + * @template {IDraggableTemplate} U + * @extends {IElement} + */ +class IDraggableElement extends IElement { + + static properties = { + ...super.properties, + locationX: { + type: Number, + attribute: false, + }, + locationY: { + type: Number, + attribute: false, + }, + sizeX: { + type: Number, + attribute: false, + }, + sizeY: { + type: Number, + attribute: false, + }, + } + static dragEventName = Configuration.dragEventName + static dragGeneralEventName = Configuration.dragGeneralEventName + + constructor() { + super(); + this.locationX = 0; + this.locationY = 0; + this.sizeX = 0; + this.sizeY = 0; + } + + computeSizes() { + const bounding = this.getBoundingClientRect(); + this.sizeX = this.blueprint.scaleCorrect(bounding.width); + this.sizeY = this.blueprint.scaleCorrect(bounding.height); + } + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + this.computeSizes(); + } + + /** + * @param {Number} x + * @param {Number} y + */ + setLocation(x, y, acknowledge = true) { + const dx = x - this.locationX; + const dy = y - this.locationY; + this.locationX = x; + this.locationY = y; + if (this.blueprint && acknowledge) { + const dragLocalEvent = new CustomEvent( + /** @type {typeof IDraggableElement} */(this.constructor).dragEventName, + { + detail: { + value: [dx, dy], + }, + bubbles: false, + cancelable: true, + } + ); + this.dispatchEvent(dragLocalEvent); + } + } + + /** + * @param {Number} x + * @param {Number} y + */ + addLocation(x, y, acknowledge = true) { + this.setLocation(this.locationX + x, this.locationY + y, acknowledge); + } + + /** @param {Number[]} value */ + acknowledgeDrag(value) { + const dragEvent = new CustomEvent( + /** @type {typeof IDraggableElement} */(this.constructor).dragGeneralEventName, + { + detail: { + value: value + }, + bubbles: true, + cancelable: true + } + ); + this.dispatchEvent(dragEvent); + } + + snapToGrid() { + const snappedLocation = Utility.snapToGrid(this.locationX, this.locationY, Configuration.gridSize); + if (this.locationX != snappedLocation[0] || this.locationY != snappedLocation[1]) { + this.setLocation(snappedLocation[0], snappedLocation[1]); + } + } + + topBoundary(justSelectableArea = false) { + return this.template.topBoundary(justSelectableArea) + } + + rightBoundary(justSelectableArea = false) { + return this.template.rightBoundary(justSelectableArea) + } + + bottomBoundary(justSelectableArea = false) { + return this.template.bottomBoundary(justSelectableArea) + } + + leftBoundary(justSelectableArea = false) { + return this.template.leftBoundary(justSelectableArea) + } +} + +/** + * @typedef {import("../../Blueprint").default} Blueprint + * @typedef {import("../../element/IElement").default} IElement + */ + +/** + * @template {IElement} T + * @extends {IPointing} + */ +class IMouseClickDrag extends IPointing { + + /** @param {MouseEvent} e */ + #mouseDownHandler = e => { + this.blueprint.setFocused(true); + switch (e.button) { + case this.options.clickButton: + // Either doesn't matter or consider the click only when clicking on the parent, not descandants + if (!this.options.strictTarget || e.target == e.currentTarget) { + if (this.options.consumeEvent) { + e.stopImmediatePropagation(); // Captured, don't call anyone else + } + // Attach the listeners + this.#movementListenedElement.addEventListener("mousemove", this.#mouseStartedMovingHandler); + document.addEventListener("mouseup", this.#mouseUpHandler); + this.clickedPosition = this.locationFromEvent(e); + this.blueprint.mousePosition[0] = this.clickedPosition[0]; + this.blueprint.mousePosition[1] = this.clickedPosition[1]; + if (this.target instanceof IDraggableElement) { + this.clickedOffset = [ + this.clickedPosition[0] - this.target.locationX, + this.clickedPosition[1] - this.target.locationY, + ]; + } + this.clicked(this.clickedPosition); + } + break + default: + if (!this.options.exitAnyButton) { + this.#mouseUpHandler(e); + } + break + } + } + + /** @param {MouseEvent} e */ + #mouseStartedMovingHandler = e => { + if (this.options.consumeEvent) { + e.stopImmediatePropagation(); // Captured, don't call anyone else + } + // Delegate from now on to this.#mouseMoveHandler + this.#movementListenedElement.removeEventListener("mousemove", this.#mouseStartedMovingHandler); + this.#movementListenedElement.addEventListener("mousemove", this.#mouseMoveHandler); + // Handler calls e.preventDefault() when it receives the event, this means dispatchEvent returns false + const dragEvent = this.getEvent(Configuration.trackingMouseEventName.begin); + this.#trackingMouse = this.target.dispatchEvent(dragEvent) == false; + const location = this.locationFromEvent(e); + // Do actual actions + this.lastLocation = Utility.snapToGrid(this.clickedPosition[0], this.clickedPosition[1], this.stepSize); + this.startDrag(location); + this.started = true; + } + + /** @param {MouseEvent} e */ + #mouseMoveHandler = e => { + if (this.options.consumeEvent) { + e.stopImmediatePropagation(); // Captured, don't call anyone else + } + const location = this.locationFromEvent(e); + const movement = [e.movementX, e.movementY]; + this.dragTo(location, movement); + if (this.#trackingMouse) { + this.blueprint.mousePosition = location; + } + if (this.options.scrollGraphEdge) { + const movementNorm = Math.sqrt(movement[0] * movement[0] + movement[1] * movement[1]); + const threshold = this.blueprint.scaleCorrect(Configuration.edgeScrollThreshold); + const leftThreshold = this.blueprint.template.gridLeftVisibilityBoundary() + threshold; + const rightThreshold = this.blueprint.template.gridRightVisibilityBoundary() - threshold; + let scrollX = 0; + if (location[0] < leftThreshold) { + scrollX = location[0] - leftThreshold; + } else if (location[0] > rightThreshold) { + scrollX = location[0] - rightThreshold; + } + const topThreshold = this.blueprint.template.gridTopVisibilityBoundary() + threshold; + const bottomThreshold = this.blueprint.template.gridBottomVisibilityBoundary() - threshold; + let scrollY = 0; + if (location[1] < topThreshold) { + scrollY = location[1] - topThreshold; + } else if (location[1] > bottomThreshold) { + scrollY = location[1] - bottomThreshold; + } + scrollX = Utility.clamp(this.blueprint.scaleCorrectReverse(scrollX) ** 3 * movementNorm * 0.6, -20, 20); + scrollY = Utility.clamp(this.blueprint.scaleCorrectReverse(scrollY) ** 3 * movementNorm * 0.6, -20, 20); + this.blueprint.scrollDelta(scrollX, scrollY); + } + } + + /** @param {MouseEvent} e */ + #mouseUpHandler = e => { + if (!this.options.exitAnyButton || e.button == this.options.clickButton) { + if (this.options.consumeEvent) { + e.stopImmediatePropagation(); // Captured, don't call anyone else + } + // Remove the handlers of "mousemove" and "mouseup" + this.#movementListenedElement.removeEventListener("mousemove", this.#mouseStartedMovingHandler); + this.#movementListenedElement.removeEventListener("mousemove", this.#mouseMoveHandler); + document.removeEventListener("mouseup", this.#mouseUpHandler); + if (this.started) { + this.endDrag(); + } + this.unclicked(); + if (this.#trackingMouse) { + const dragEvent = this.getEvent(Configuration.trackingMouseEventName.end); + this.target.dispatchEvent(dragEvent); + this.#trackingMouse = false; + } + this.started = false; + } + } + + #trackingMouse = false + #movementListenedElement + #draggableElement + + clickedOffset = [0, 0] + clickedPosition = [0, 0] + lastLocation = [0, 0] + started = false + stepSize = 1 + + /** + * @param {T} target + * @param {Blueprint} blueprint + * @param {Object} options + */ + constructor(target, blueprint, options = {}) { + options.clickButton ??= 0; + options.consumeEvent ??= true; + options.draggableElement ??= target; + options.exitAnyButton ??= true; + options.moveEverywhere ??= false; + options.movementSpace ??= blueprint?.getGridDOMElement(); + options.repositionOnClick ??= false; + options.scrollGraphEdge ??= false; + options.strictTarget ??= false; + super(target, blueprint, options); + this.stepSize = parseInt(options?.stepSize ?? Configuration.gridSize); + this.#movementListenedElement = this.options.moveEverywhere ? document.documentElement : this.movementSpace; + this.#draggableElement = /** @type {HTMLElement} */(this.options.draggableElement); + + this.listenEvents(); + } + + listenEvents() { + super.listenEvents(); + this.#draggableElement.addEventListener("mousedown", this.#mouseDownHandler); + if (this.options.clickButton == 2) { + this.#draggableElement.addEventListener("contextmenu", e => e.preventDefault()); + } + } + + unlistenEvents() { + super.unlistenEvents(); + this.#draggableElement.removeEventListener("mousedown", this.#mouseDownHandler); + } + + getEvent(eventName) { + return new CustomEvent(eventName, { + detail: { + tracker: this + }, + bubbles: true, + cancelable: true + }) + } + + /* Subclasses will override the following methods */ + clicked(location) { + } + + startDrag(location) { + } + + dragTo(location, offset) { + } + + endDrag() { + } + + unclicked(location) { + } +} + class MouseScrollGraph extends IMouseClickDrag { startDrag() { @@ -4725,8 +4765,8 @@ class MouseScrollGraph extends IMouseClickDrag { endDrag() { this.blueprint.scrolling = false; } -} - +} + class MouseTracking extends IPointing { /** @type {IPointing} */ @@ -4789,32 +4829,32 @@ class MouseTracking extends IPointing { /** @type {(e: Event) => any} */(this.#trackingMouseGaveBackHandler) ); } -} - -/** - * @typedef {import("./IElement").default} IElement - * @typedef {new (...args) => IElement} ElementConstructor - */ - -class ElementFactory { - - /** @type {Map} */ - static #elementConstructors = new Map() - - /** - * @param {String} tagName - * @param {ElementConstructor} entityConstructor - */ - static registerElement(tagName, entityConstructor) { - ElementFactory.#elementConstructors.set(tagName, entityConstructor); - } - - /** @param {String} tagName */ - static getConstructor(tagName) { - return ElementFactory.#elementConstructors.get(tagName) - } -} - +} + +/** + * @typedef {import("./IElement").default} IElement + * @typedef {new (...args) => IElement} ElementConstructor + */ + +class ElementFactory { + + /** @type {Map} */ + static #elementConstructors = new Map() + + /** + * @param {String} tagName + * @param {ElementConstructor} entityConstructor + */ + static registerElement(tagName, entityConstructor) { + ElementFactory.#elementConstructors.set(tagName, entityConstructor); + } + + /** @param {String} tagName */ + static getConstructor(tagName) { + return ElementFactory.#elementConstructors.get(tagName) + } +} + /** @typedef {import("../../element/NodeElement").NodeElementConstructor} NodeElementConstructor */ class Paste extends IInput { @@ -4867,8 +4907,8 @@ class Paste extends IInput { this.blueprint.addGraphElement(...nodes); return true } -} - +} + class Select extends IMouseClickDrag { constructor(target, blueprint, options = {}) { @@ -4896,8 +4936,8 @@ class Select extends IMouseClickDrag { this.blueprint.unselectAll(); } } -} - +} + class Unfocus extends IInput { /** @param {MouseEvent} e */ @@ -4927,8 +4967,8 @@ class Unfocus extends IInput { unlistenEvents() { document.removeEventListener("click", this.#clickHandler); } -} - +} + /** * @typedef {import("../Blueprint").default} Blueprint * @typedef {import("../element/PinElement").default} PinElement @@ -5155,75 +5195,75 @@ class BlueprintTemplate extends ITemplate { avgY = nodes.length > 0 ? Math.round(avgY / (2 * nodes.length)) : 0; this.centerViewport(avgX, avgY, smooth); } -} - -/** - * @typedef {import("../entity/IEntity").default} IEntity - * @typedef {import("../template/ITemplate").default} ITemplate - */ - -/** - * @template {IEntity} T - * @template {ITemplate} U - * @extends {IElement} - */ -class IFromToPositionedElement extends IElement { - - static properties = { - ...super.properties, - fromX: { - type: Number, - attribute: false, - }, - fromY: { - type: Number, - attribute: false, - }, - toX: { - type: Number, - attribute: false, - }, - toY: { - type: Number, - attribute: false, - }, - } - - constructor() { - super(); - this.fromX = 0; - this.fromY = 0; - this.toX = 0; - this.toY = 0; - } - - /** @param {Number[]} param0 */ - setBothLocations([x, y]) { - this.fromX = x; - this.fromY = y; - this.toX = x; - this.toY = y; - } - - /** - * @param {Number} x - * @param {Number} y - */ - addSourceLocation(x, y) { - this.fromX += x; - this.fromY += y; - } - - /** - * @param {Number} x - * @param {Number} y - */ - addDestinationLocation(x, y) { - this.toX += x; - this.toY += y; - } -} - +} + +/** + * @typedef {import("../entity/IEntity").default} IEntity + * @typedef {import("../template/ITemplate").default} ITemplate + */ + +/** + * @template {IEntity} T + * @template {ITemplate} U + * @extends {IElement} + */ +class IFromToPositionedElement extends IElement { + + static properties = { + ...super.properties, + fromX: { + type: Number, + attribute: false, + }, + fromY: { + type: Number, + attribute: false, + }, + toX: { + type: Number, + attribute: false, + }, + toY: { + type: Number, + attribute: false, + }, + } + + constructor() { + super(); + this.fromX = 0; + this.fromY = 0; + this.toX = 0; + this.toY = 0; + } + + /** @param {Number[]} param0 */ + setBothLocations([x, y]) { + this.fromX = x; + this.fromY = y; + this.toX = x; + this.toY = y; + } + + /** + * @param {Number} x + * @param {Number} y + */ + addSourceLocation(x, y) { + this.fromX += x; + this.fromY += y; + } + + /** + * @param {Number} x + * @param {Number} y + */ + addDestinationLocation(x, y) { + this.toX += x; + this.toY += y; + } +} + /** * @typedef {import("../element/IFromToPositionedElement").default} IFromToPositionedElement * @typedef {import("lit").PropertyValues} PropertyValues @@ -5259,95 +5299,95 @@ class IFromToPositionedTemplate extends ITemplate { this.element.style.height = `${height}px`; } } -} - -class KnotEntity extends ObjectEntity { - - /** - * @param {Object} options - * @param {PinEntity} pinReferenceForType - */ - constructor(options = {}, pinReferenceForType = undefined) { - super(options, true); - this.Class = new ObjectReferenceEntity("/Script/BlueprintGraph.K2Node_Knot"); - this.Name = "K2Node_Knot"; - const inputPinEntity = new PinEntity( - { - PinName: "InputPin", - }, - true - ); - const outputPinEntity = new PinEntity( - { - PinName: "OutputPin", - Direction: "EGPD_Output", - }, - true - ); - if (pinReferenceForType) { - inputPinEntity.copyTypeFrom(pinReferenceForType); - outputPinEntity.copyTypeFrom(pinReferenceForType); - } - this.CustomProperties = [inputPinEntity, outputPinEntity]; - } -} - -/** @typedef {import("../../Blueprint").default} Blueprint */ - -/** - * @template {HTMLElement} T - * @extends {IPointing} - */ -class MouseDbClick extends IPointing { - - /** @param {Number[]} location */ - static ignoreDbClick = location => { } - - /** @param {MouseEvent} e */ - #mouseDbClickHandler = e => { - if (!this.options.strictTarget || e.target === e.currentTarget) { - if (this.options.consumeEvent) { - e.stopImmediatePropagation(); // Captured, don't call anyone else - } - this.clickedPosition = this.locationFromEvent(e); - this.blueprint.mousePosition[0] = this.clickedPosition[0]; - this.blueprint.mousePosition[1] = this.clickedPosition[1]; - this.dbclicked(this.clickedPosition); - } - } - - #onDbClick - get onDbClick() { - return this.#onDbClick - } - set onDbClick(value) { - this.#onDbClick = value; - } - - clickedPosition = [0, 0] - - constructor(target, blueprint, options = {}, onDbClick = MouseDbClick.ignoreDbClick) { - options.consumeEvent ??= true; - options.strictTarget ??= false; - super(target, blueprint, options); - this.#onDbClick = onDbClick; - this.listenEvents(); - } - - listenEvents() { - this.target.addEventListener("dblclick", this.#mouseDbClickHandler); - } - - unlistenEvents() { - this.target.removeEventListener("dblclick", this.#mouseDbClickHandler); - } - - /* Subclasses will override the following method */ - dbclicked(location) { - this.onDbClick(location); - } -} - +} + +class KnotEntity extends ObjectEntity { + + /** + * @param {Object} options + * @param {PinEntity} pinReferenceForType + */ + constructor(options = {}, pinReferenceForType = undefined) { + super(options, true); + this.Class = new ObjectReferenceEntity("/Script/BlueprintGraph.K2Node_Knot"); + this.Name = "K2Node_Knot"; + const inputPinEntity = new PinEntity( + { + PinName: "InputPin", + }, + true + ); + const outputPinEntity = new PinEntity( + { + PinName: "OutputPin", + Direction: "EGPD_Output", + }, + true + ); + if (pinReferenceForType) { + inputPinEntity.copyTypeFrom(pinReferenceForType); + outputPinEntity.copyTypeFrom(pinReferenceForType); + } + this.CustomProperties = [inputPinEntity, outputPinEntity]; + } +} + +/** @typedef {import("../../Blueprint").default} Blueprint */ + +/** + * @template {HTMLElement} T + * @extends {IPointing} + */ +class MouseDbClick extends IPointing { + + /** @param {Number[]} location */ + static ignoreDbClick = location => { } + + /** @param {MouseEvent} e */ + #mouseDbClickHandler = e => { + if (!this.options.strictTarget || e.target === e.currentTarget) { + if (this.options.consumeEvent) { + e.stopImmediatePropagation(); // Captured, don't call anyone else + } + this.clickedPosition = this.locationFromEvent(e); + this.blueprint.mousePosition[0] = this.clickedPosition[0]; + this.blueprint.mousePosition[1] = this.clickedPosition[1]; + this.dbclicked(this.clickedPosition); + } + } + + #onDbClick + get onDbClick() { + return this.#onDbClick + } + set onDbClick(value) { + this.#onDbClick = value; + } + + clickedPosition = [0, 0] + + constructor(target, blueprint, options = {}, onDbClick = MouseDbClick.ignoreDbClick) { + options.consumeEvent ??= true; + options.strictTarget ??= false; + super(target, blueprint, options); + this.#onDbClick = onDbClick; + this.listenEvents(); + } + + listenEvents() { + this.target.addEventListener("dblclick", this.#mouseDbClickHandler); + } + + unlistenEvents() { + this.target.removeEventListener("dblclick", this.#mouseDbClickHandler); + } + + /* Subclasses will override the following method */ + dbclicked(location) { + this.onDbClick(location); + } +} + /** * @typedef {import("../element/LinkElement").default} LinkElement * @typedef {import("../element/LinkElement").LinkElementConstructor} LinkElementConstructor @@ -5532,313 +5572,313 @@ class LinkTemplate extends IFromToPositionedTemplate { ` : b} ` } -} - -/** - * @typedef {import("../element/IDraggableElement").DragEvent} DragEvent - * @typedef {import("./PinElement").default} PinElement - * @typedef {import("lit").TemplateResult<1>} TemplateResult - * @typedef {typeof LinkElement} LinkElementConstructor - */ - -/** @extends {IFromToPositionedElement} */ -class LinkElement extends IFromToPositionedElement { - - static properties = { - ...super.properties, - source: { - type: String, - reflect: true, - }, - destination: { - type: String, - reflect: true, - }, - dragging: { - type: Boolean, - attribute: "data-dragging", - converter: Utility.booleanConverter, - reflect: true, - }, - originatesFromInput: { - type: Boolean, - attribute: false, - }, - svgPathD: { - type: String, - attribute: false, - }, - linkMessageIcon: { - type: String, - attribute: false, - }, - linkMessageText: { - type: String, - attribute: false, - }, - } - - /** @type {PinElement} */ - #sourcePin - get sourcePin() { - return this.#sourcePin - } - set sourcePin(pin) { - this.#setPin(pin, false); - } - - /** @type {PinElement} */ - #destinationPin - get destinationPin() { - return this.#destinationPin - } - set destinationPin(pin) { - this.#setPin(pin, true); - } - - #nodeDeleteHandler = () => this.remove() - /** @param {DragEvent} e */ - #nodeDragSourceHandler = e => this.addSourceLocation(...e.detail.value) - /** @param {DragEvent} e */ - #nodeDragDestinatonHandler = e => this.addDestinationLocation(...e.detail.value) - #nodeReflowSourceHandler = e => this.setSourceLocation() - #nodeReflowDestinatonHandler = e => this.setDestinationLocation() - - /** @type {TemplateResult | nothing} */ - linkMessageIcon = b - /** @type {TemplateResult | nothing} */ - linkMessageText = b - - /** @type {SVGPathElement} */ - pathElement - - constructor() { - super(); - this.source = null; - this.destination = null; - this.dragging = false; - this.originatesFromInput = false; - this.startPercentage = 0; - this.svgPathD = ""; - this.startPixels = 0; - } - - /** - * @param {PinElement} source - * @param {PinElement?} destination - */ - static newObject(source, destination) { - const result = new LinkElement(); - result.initialize(source, destination); - return result - } - - /** - * @param {PinElement} source - * @param {PinElement?} destination - */ - initialize(source, destination) { - super.initialize({}, new LinkTemplate()); - if (source) { - this.sourcePin = source; - if (!destination) { - this.toX = this.fromX; - this.toY = this.fromY; - } - } - if (destination) { - this.destinationPin = destination; - if (!source) { - this.fromX = this.toX; - this.fromY = this.toY; - } - } - } - - /** - * @param {PinElement} pin - * @param {Boolean} isDestinationPin - */ - #setPin(pin, isDestinationPin) { - const getCurrentPin = () => isDestinationPin ? this.destinationPin : this.sourcePin; - if (getCurrentPin() == pin) { - return - } - if (getCurrentPin()) { - const nodeElement = getCurrentPin().getNodeElement(); - nodeElement.removeEventListener(Configuration.removeEventName, this.#nodeDeleteHandler); - nodeElement.removeEventListener( - Configuration.nodeDragEventName, - isDestinationPin ? this.#nodeDragDestinatonHandler : this.#nodeDragSourceHandler - ); - nodeElement.removeEventListener( - Configuration.nodeReflowEventName, - isDestinationPin ? this.#nodeReflowDestinatonHandler : this.#nodeReflowSourceHandler - ); - this.#unlinkPins(); - } - isDestinationPin - ? this.#destinationPin = pin - : this.#sourcePin = pin; - if (getCurrentPin()) { - const nodeElement = getCurrentPin().getNodeElement(); - nodeElement.addEventListener(Configuration.removeEventName, this.#nodeDeleteHandler); - nodeElement.addEventListener( - Configuration.nodeDragEventName, - isDestinationPin ? this.#nodeDragDestinatonHandler : this.#nodeDragSourceHandler - ); - nodeElement.addEventListener( - Configuration.nodeReflowEventName, - isDestinationPin ? this.#nodeReflowDestinatonHandler : this.#nodeReflowSourceHandler - ); - isDestinationPin - ? this.setDestinationLocation() - : (this.setSourceLocation(), this.originatesFromInput = this.sourcePin.isInput()); - this.#linkPins(); - } - } - - #linkPins() { - if (this.sourcePin && this.destinationPin) { - this.sourcePin.linkTo(this.destinationPin); - this.destinationPin.linkTo(this.sourcePin); - } - } - - #unlinkPins() { - if (this.sourcePin && this.destinationPin) { - this.sourcePin.unlinkFrom(this.destinationPin, false); - this.destinationPin.unlinkFrom(this.sourcePin, false); - } - } - - cleanup() { - super.cleanup(); - this.#unlinkPins(); - this.sourcePin = null; - this.destinationPin = null; - } - - /** @param {Number[]?} location */ - setSourceLocation(location = null, canPostpone = true) { - if (location == null) { - const self = this; - if (canPostpone && (!this.hasUpdated || !this.sourcePin.hasUpdated)) { - Promise.all([this.updateComplete, this.sourcePin.updateComplete]) - .then(() => self.setSourceLocation(null, false)); - return - } - location = this.sourcePin.template.getLinkLocation(); - } - const [x, y] = location; - this.fromX = x; - this.fromY = y; - } - - /** @param {Number[]?} location */ - setDestinationLocation(location = null, canPostpone = true) { - if (location == null) { - const self = this; - if (canPostpone && (!this.hasUpdated || !this.destinationPin.hasUpdated)) { - Promise.all([this.updateComplete, this.destinationPin.updateComplete]) - .then(() => self.setDestinationLocation(null, false)); - return - } - location = this.destinationPin.template.getLinkLocation(); - } - this.toX = location[0]; - this.toY = location[1]; - } - - getInputPin() { - if (this.sourcePin?.isInput()) { - return this.sourcePin - } - return this.destinationPin - } - - /** @param {PinElement} pin */ - setInputPin(pin) { - if (this.sourcePin?.isInput()) { - this.sourcePin = pin; - } - this.destinationPin = pin; - } - - getOutputPin() { - if (this.destinationPin?.isOutput()) { - return this.destinationPin - } - return this.sourcePin - } - - /** @param {PinElement} pin */ - setOutputPin(pin) { - if (this.destinationPin?.isOutput()) { - this.destinationPin = pin; - } - this.sourcePin = pin; - } - - startDragging() { - this.dragging = true; - } - - finishDragging() { - this.dragging = false; - } - - removeMessage() { - this.linkMessageIcon = b; - this.linkMessageText = b; - } - - setMessageConvertType() { - this.linkMessageIcon = "ueb-icon-conver-type"; - this.linkMessageText = `Convert ${this.sourcePin.pinType} to ${this.destinationPin.pinType}.`; - } - - setMessageCorrect() { - this.linkMessageIcon = SVGIcon.correct; - this.linkMessageText = b; - } - - setMessageReplace() { - this.linkMessageIcon = SVGIcon.correct; - this.linkMessageText = b; - } - - setMessageDirectionsIncompatible() { - this.linkMessageIcon = SVGIcon.reject; - this.linkMessageText = y`Directions are not compatbile.`; - } - - setMessagePlaceNode() { - this.linkMessageIcon = "ueb-icon-place-node"; - this.linkMessageText = y`Place a new node.`; - } - - setMessageReplaceLink() { - this.linkMessageIcon = SVGIcon.correct; - this.linkMessageText = y`Replace existing input connections.`; - } - - setMessageReplaceOutputLink() { - this.linkMessageIcon = SVGIcon.correct; - this.linkMessageText = y`Replace existing output connections.`; - } - - setMessageSameNode() { - this.linkMessageIcon = SVGIcon.reject; - this.linkMessageText = y`Both are on the same node.`; - } - - setMEssagetypesIncompatible() { - this.linkMessageIcon = SVGIcon.reject; - this.linkMessageText = y`${this.sourcePin.pinType} is not compatible with ${this.destinationPin.pinType}.`; - } -} - +} + +/** + * @typedef {import("../element/IDraggableElement").DragEvent} DragEvent + * @typedef {import("./PinElement").default} PinElement + * @typedef {import("lit").TemplateResult<1>} TemplateResult + * @typedef {typeof LinkElement} LinkElementConstructor + */ + +/** @extends {IFromToPositionedElement} */ +class LinkElement extends IFromToPositionedElement { + + static properties = { + ...super.properties, + source: { + type: String, + reflect: true, + }, + destination: { + type: String, + reflect: true, + }, + dragging: { + type: Boolean, + attribute: "data-dragging", + converter: Utility.booleanConverter, + reflect: true, + }, + originatesFromInput: { + type: Boolean, + attribute: false, + }, + svgPathD: { + type: String, + attribute: false, + }, + linkMessageIcon: { + type: String, + attribute: false, + }, + linkMessageText: { + type: String, + attribute: false, + }, + } + + /** @type {PinElement} */ + #sourcePin + get sourcePin() { + return this.#sourcePin + } + set sourcePin(pin) { + this.#setPin(pin, false); + } + + /** @type {PinElement} */ + #destinationPin + get destinationPin() { + return this.#destinationPin + } + set destinationPin(pin) { + this.#setPin(pin, true); + } + + #nodeDeleteHandler = () => this.remove() + /** @param {DragEvent} e */ + #nodeDragSourceHandler = e => this.addSourceLocation(...e.detail.value) + /** @param {DragEvent} e */ + #nodeDragDestinatonHandler = e => this.addDestinationLocation(...e.detail.value) + #nodeReflowSourceHandler = e => this.setSourceLocation() + #nodeReflowDestinatonHandler = e => this.setDestinationLocation() + + /** @type {TemplateResult | nothing} */ + linkMessageIcon = b + /** @type {TemplateResult | nothing} */ + linkMessageText = b + + /** @type {SVGPathElement} */ + pathElement + + constructor() { + super(); + this.source = null; + this.destination = null; + this.dragging = false; + this.originatesFromInput = false; + this.startPercentage = 0; + this.svgPathD = ""; + this.startPixels = 0; + } + + /** + * @param {PinElement} source + * @param {PinElement?} destination + */ + static newObject(source, destination) { + const result = new LinkElement(); + result.initialize(source, destination); + return result + } + + /** + * @param {PinElement} source + * @param {PinElement?} destination + */ + initialize(source, destination) { + super.initialize({}, new LinkTemplate()); + if (source) { + this.sourcePin = source; + if (!destination) { + this.toX = this.fromX; + this.toY = this.fromY; + } + } + if (destination) { + this.destinationPin = destination; + if (!source) { + this.fromX = this.toX; + this.fromY = this.toY; + } + } + } + + /** + * @param {PinElement} pin + * @param {Boolean} isDestinationPin + */ + #setPin(pin, isDestinationPin) { + const getCurrentPin = () => isDestinationPin ? this.destinationPin : this.sourcePin; + if (getCurrentPin() == pin) { + return + } + if (getCurrentPin()) { + const nodeElement = getCurrentPin().getNodeElement(); + nodeElement.removeEventListener(Configuration.removeEventName, this.#nodeDeleteHandler); + nodeElement.removeEventListener( + Configuration.nodeDragEventName, + isDestinationPin ? this.#nodeDragDestinatonHandler : this.#nodeDragSourceHandler + ); + nodeElement.removeEventListener( + Configuration.nodeReflowEventName, + isDestinationPin ? this.#nodeReflowDestinatonHandler : this.#nodeReflowSourceHandler + ); + this.#unlinkPins(); + } + isDestinationPin + ? this.#destinationPin = pin + : this.#sourcePin = pin; + if (getCurrentPin()) { + const nodeElement = getCurrentPin().getNodeElement(); + nodeElement.addEventListener(Configuration.removeEventName, this.#nodeDeleteHandler); + nodeElement.addEventListener( + Configuration.nodeDragEventName, + isDestinationPin ? this.#nodeDragDestinatonHandler : this.#nodeDragSourceHandler + ); + nodeElement.addEventListener( + Configuration.nodeReflowEventName, + isDestinationPin ? this.#nodeReflowDestinatonHandler : this.#nodeReflowSourceHandler + ); + isDestinationPin + ? this.setDestinationLocation() + : (this.setSourceLocation(), this.originatesFromInput = this.sourcePin.isInput()); + this.#linkPins(); + } + } + + #linkPins() { + if (this.sourcePin && this.destinationPin) { + this.sourcePin.linkTo(this.destinationPin); + this.destinationPin.linkTo(this.sourcePin); + } + } + + #unlinkPins() { + if (this.sourcePin && this.destinationPin) { + this.sourcePin.unlinkFrom(this.destinationPin, false); + this.destinationPin.unlinkFrom(this.sourcePin, false); + } + } + + cleanup() { + super.cleanup(); + this.#unlinkPins(); + this.sourcePin = null; + this.destinationPin = null; + } + + /** @param {Number[]?} location */ + setSourceLocation(location = null, canPostpone = true) { + if (location == null) { + const self = this; + if (canPostpone && (!this.hasUpdated || !this.sourcePin.hasUpdated)) { + Promise.all([this.updateComplete, this.sourcePin.updateComplete]) + .then(() => self.setSourceLocation(null, false)); + return + } + location = this.sourcePin.template.getLinkLocation(); + } + const [x, y] = location; + this.fromX = x; + this.fromY = y; + } + + /** @param {Number[]?} location */ + setDestinationLocation(location = null, canPostpone = true) { + if (location == null) { + const self = this; + if (canPostpone && (!this.hasUpdated || !this.destinationPin.hasUpdated)) { + Promise.all([this.updateComplete, this.destinationPin.updateComplete]) + .then(() => self.setDestinationLocation(null, false)); + return + } + location = this.destinationPin.template.getLinkLocation(); + } + this.toX = location[0]; + this.toY = location[1]; + } + + getInputPin() { + if (this.sourcePin?.isInput()) { + return this.sourcePin + } + return this.destinationPin + } + + /** @param {PinElement} pin */ + setInputPin(pin) { + if (this.sourcePin?.isInput()) { + this.sourcePin = pin; + } + this.destinationPin = pin; + } + + getOutputPin() { + if (this.destinationPin?.isOutput()) { + return this.destinationPin + } + return this.sourcePin + } + + /** @param {PinElement} pin */ + setOutputPin(pin) { + if (this.destinationPin?.isOutput()) { + this.destinationPin = pin; + } + this.sourcePin = pin; + } + + startDragging() { + this.dragging = true; + } + + finishDragging() { + this.dragging = false; + } + + removeMessage() { + this.linkMessageIcon = b; + this.linkMessageText = b; + } + + setMessageConvertType() { + this.linkMessageIcon = "ueb-icon-conver-type"; + this.linkMessageText = `Convert ${this.sourcePin.pinType} to ${this.destinationPin.pinType}.`; + } + + setMessageCorrect() { + this.linkMessageIcon = SVGIcon.correct; + this.linkMessageText = b; + } + + setMessageReplace() { + this.linkMessageIcon = SVGIcon.correct; + this.linkMessageText = b; + } + + setMessageDirectionsIncompatible() { + this.linkMessageIcon = SVGIcon.reject; + this.linkMessageText = y`Directions are not compatbile.`; + } + + setMessagePlaceNode() { + this.linkMessageIcon = "ueb-icon-place-node"; + this.linkMessageText = y`Place a new node.`; + } + + setMessageReplaceLink() { + this.linkMessageIcon = SVGIcon.correct; + this.linkMessageText = y`Replace existing input connections.`; + } + + setMessageReplaceOutputLink() { + this.linkMessageIcon = SVGIcon.correct; + this.linkMessageText = y`Replace existing output connections.`; + } + + setMessageSameNode() { + this.linkMessageIcon = SVGIcon.reject; + this.linkMessageText = y`Both are on the same node.`; + } + + setMEssagetypesIncompatible() { + this.linkMessageIcon = SVGIcon.reject; + this.linkMessageText = y`${this.sourcePin.pinType} is not compatible with ${this.destinationPin.pinType}.`; + } +} + /** * @typedef {import("../../Blueprint").default} Blueprint * @typedef {import("../../element/IDraggableElement").default} IDraggableElement @@ -5898,8 +5938,8 @@ class MouseMoveDraggable extends IMouseClickDrag { dragAction(location, offset) { this.target.setLocation(location[0] - this.clickedOffset[0], location[1] - this.clickedOffset[1]); } -} - +} + /** @typedef {import("../../Blueprint").default} Blueprint */ class MouseClickDrag extends MouseMoveDraggable { @@ -5949,8 +5989,8 @@ class MouseClickDrag extends MouseMoveDraggable { super.endDrag(); this.#onEndDrag?.(); } -} - +} + /** * @typedef {import("../entity/IEntity").default} IEntity * @typedef {import("../element/IDraggableElement").default} IDraggableElement @@ -6008,31 +6048,31 @@ class IDraggableTemplate extends ITemplate { let avgY = Math.max((dt + db) / 2, minMargin); this.blueprint.scrollDelta(dl - avgX, dt - avgY, true); } -} - -/** - * @typedef {import("../element/IDraggableElement").default} IDraggableElement - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -/** - * @template {IDraggableElement} T - * @extends {IDraggableTemplate} - */ -class IDraggablePositionedTemplate extends IDraggableTemplate { - - /** @param {PropertyValues} changedProperties */ - update(changedProperties) { - super.update(changedProperties); - if (changedProperties.has("locationX")) { - this.element.style.left = `${this.element.locationX}px`; - } - if (changedProperties.has("locationY")) { - this.element.style.top = `${this.element.locationY}px`; - } - } -} - +} + +/** + * @typedef {import("../element/IDraggableElement").default} IDraggableElement + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +/** + * @template {IDraggableElement} T + * @extends {IDraggableTemplate} + */ +class IDraggablePositionedTemplate extends IDraggableTemplate { + + /** @param {PropertyValues} changedProperties */ + update(changedProperties) { + super.update(changedProperties); + if (changedProperties.has("locationX")) { + this.element.style.left = `${this.element.locationX}px`; + } + if (changedProperties.has("locationY")) { + this.element.style.top = `${this.element.locationY}px`; + } + } +} + /** * @typedef {import("../../Blueprint").default} Blueprint * @typedef {import("../../element/NodeElement").default} NodeElement @@ -6068,8 +6108,8 @@ class MouseMoveNodes extends MouseMoveDraggable { ); } } -} - +} + /** * @typedef {import("../element/NodeElement").default} NodeElement * @typedef {import("lit").PropertyValues} PropertyValues @@ -6100,8 +6140,8 @@ class ISelectableDraggableTemplate extends IDraggablePositionedTemplate { this.element.setSelected(true); } } -} - +} + /** * @typedef {import("../../element/NodeElement").default} NodeElement * @typedef {import("../../element/PinElement").default} PinElement @@ -6242,275 +6282,275 @@ class NodeTemplate extends ISelectableDraggableTemplate { } linksChanged() { } -} - -/** - * @typedef {import("../element/NodeElement").default} NodeElement - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -class IResizeableTemplate extends NodeTemplate { - - #THandler = document.createElement("div") - #RHandler = document.createElement("div") - #BHandler = document.createElement("div") - #LHandler = document.createElement("div") - #TRHandler = document.createElement("div") - #BRHandler = document.createElement("div") - #BLHandler = document.createElement("div") - #TLHandler = document.createElement("div") - - /** @param {NodeElement} element */ - initialize(element) { - super.initialize(element); - this.element.classList.add("ueb-resizeable"); - this.#THandler.classList.add("ueb-resizeable-top"); - this.#RHandler.classList.add("ueb-resizeable-right"); - this.#BHandler.classList.add("ueb-resizeable-bottom"); - this.#LHandler.classList.add("ueb-resizeable-left"); - this.#TRHandler.classList.add("ueb-resizeable-top-right"); - this.#BRHandler.classList.add("ueb-resizeable-bottom-right"); - this.#BLHandler.classList.add("ueb-resizeable-bottom-left"); - this.#TLHandler.classList.add("ueb-resizeable-top-left"); - } - - /** @param {PropertyValues} changedProperties */ - update(changedProperties) { - super.update(changedProperties); - if (this.element.sizeX >= 0 && changedProperties.has("sizeX")) { - this.element.style.width = `${this.element.sizeX}px`; - } - if (this.element.sizeY >= 0 && changedProperties.has("sizeY")) { - this.element.style.height = `${this.element.sizeY}px`; - } - } - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - this.element.append( - this.#THandler, - this.#RHandler, - this.#BHandler, - this.#LHandler, - this.#TRHandler, - this.#BRHandler, - this.#BLHandler, - this.#TLHandler - ); - } - - createInputObjects() { - return [ - ...super.createInputObjects(), - new MouseClickDrag(this.#THandler, this.blueprint, { - onDrag: (location, movement) => { - movement[1] = location[1] - this.element.topBoundary(); - if (this.setSizeY(this.element.sizeY - movement[1])) { - this.element.addLocation(0, movement[1], false); - } - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#RHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[0] = location[0] - this.element.rightBoundary(); - this.setSizeX(this.element.sizeX + movement[0]); - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#BHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[1] = location[1] - this.element.bottomBoundary(); - this.setSizeY(this.element.sizeY + movement[1]); - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#LHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[0] = location[0] - this.element.leftBoundary(); - if (this.setSizeX(this.element.sizeX - movement[0])) { - this.element.addLocation(movement[0], 0, false); - } - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#TRHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[0] = location[0] - this.element.rightBoundary(); - movement[1] = location[1] - this.element.topBoundary(); - this.setSizeX(this.element.sizeX + movement[0]); - if (this.setSizeY(this.element.sizeY - movement[1])) { - this.element.addLocation(0, movement[1], false); - } - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#BRHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[0] = location[0] - this.element.rightBoundary(); - movement[1] = location[1] - this.element.bottomBoundary(); - this.setSizeX(this.element.sizeX + movement[0]); - this.setSizeY(this.element.sizeY + movement[1]); - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#BLHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[0] = location[0] - this.element.leftBoundary(); - movement[1] = location[1] - this.element.bottomBoundary(); - if (this.setSizeX(this.element.sizeX - movement[0])) { - this.element.addLocation(movement[0], 0, false); - } - this.setSizeY(this.element.sizeY + movement[1]); - }, - onEndDrag: () => this.endResize(), - }), - new MouseClickDrag(this.#TLHandler, this.blueprint, { - onDrag: (location, movement) => { - movement[0] = location[0] - this.element.leftBoundary(); - movement[1] = location[1] - this.element.topBoundary(); - if (this.setSizeX(this.element.sizeX - movement[0])) { - this.element.addLocation(movement[0], 0, false); - } - if (this.setSizeY(this.element.sizeY - movement[1])) { - this.element.addLocation(0, movement[1], false); - } - }, - onEndDrag: () => this.endResize(), - }), - ] - } - - /** @param {Number} value */ - setSizeX(value) { - this.element.setNodeWidth(value); - return true - } - - /** @param {Number} value */ - setSizeY(value) { - this.element.setNodeHeight(value); - return true - } - - endResize() { - } -} - -/** - * @typedef {import("../../element/NodeElement").default} NodeElement - * @typedef {import("../../element/PinElement").default} PinElement - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -class CommentNodeTemplate extends IResizeableTemplate { - - #color = LinearColorEntity.getWhite() - #selectableAreaHeight = 0 - - /** @param {NodeElement} element */ - initialize(element) { - if (element.entity.CommentColor) { - this.#color.setFromRGBANumber(element.entity.CommentColor.toNumber()); - this.#color.setFromHSVA( - this.#color.H.value, - this.#color.S.value, - Math.pow(this.#color.V.value, 0.45) * 0.67 - ); - } - element.classList.add("ueb-node-style-comment", "ueb-node-resizeable"); - element.sizeX = 25 * Configuration.gridSize; - element.sizeY = 6 * Configuration.gridSize; - super.initialize(element); // Keep it at the end because it calls this.getColor() where this.#color must be initialized - } - - getColor() { - return i$3`${Math.round(this.#color.R.value * 255)}, ${Math.round(this.#color.G.value * 255)}, ${Math.round(this.#color.B.value * 255)}` - } - - getDraggableElement() { - return this.element.querySelector(".ueb-node-top") - } - - render() { - return y` -
-
-
- ${this.element.entity.NodeComment} -
-
-
- ` - } - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - const bounding = this.getDraggableElement().getBoundingClientRect(); - this.#selectableAreaHeight = bounding.height; - } - - manageNodesBind() { - let nodes = this.blueprint.getNodes(); - for (let node of nodes) { - if ( - node.topBoundary() >= this.element.topBoundary() - && node.rightBoundary() <= this.element.rightBoundary() - && node.bottomBoundary() <= this.element.bottomBoundary() - && node.leftBoundary() >= this.element.leftBoundary() - ) { - node.bindToComment(this.element); - } else { - node.unbindFromComment(this.element); - } - } - } - - /** @param {Number} value */ - setSizeX(value) { - value = Math.round(value); - if (value >= Configuration.gridSet * Configuration.gridSize) { - this.element.setNodeWidth(value); - return true - } - return false - } - - /** @param {Number} value */ - setSizeY(value) { - value = Math.round(value); - if (value >= 3 * Configuration.gridSize) { - this.element.setNodeHeight(value); - return true - } - return false - } - - endResize() { - this.manageNodesBind(); - } - - topBoundary(justSelectableArea = false) { - return this.element.locationY - } - - rightBoundary(justSelectableArea = false) { - return this.element.locationX + this.element.sizeX - } - - bottomBoundary(justSelectableArea = false) { - return justSelectableArea - ? this.element.locationY + this.#selectableAreaHeight - : super.bottomBoundary() - } - - leftBoundary(justSelectableArea = false) { - return this.element.locationX - } -} - +} + +/** + * @typedef {import("../element/NodeElement").default} NodeElement + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +class IResizeableTemplate extends NodeTemplate { + + #THandler = document.createElement("div") + #RHandler = document.createElement("div") + #BHandler = document.createElement("div") + #LHandler = document.createElement("div") + #TRHandler = document.createElement("div") + #BRHandler = document.createElement("div") + #BLHandler = document.createElement("div") + #TLHandler = document.createElement("div") + + /** @param {NodeElement} element */ + initialize(element) { + super.initialize(element); + this.element.classList.add("ueb-resizeable"); + this.#THandler.classList.add("ueb-resizeable-top"); + this.#RHandler.classList.add("ueb-resizeable-right"); + this.#BHandler.classList.add("ueb-resizeable-bottom"); + this.#LHandler.classList.add("ueb-resizeable-left"); + this.#TRHandler.classList.add("ueb-resizeable-top-right"); + this.#BRHandler.classList.add("ueb-resizeable-bottom-right"); + this.#BLHandler.classList.add("ueb-resizeable-bottom-left"); + this.#TLHandler.classList.add("ueb-resizeable-top-left"); + } + + /** @param {PropertyValues} changedProperties */ + update(changedProperties) { + super.update(changedProperties); + if (this.element.sizeX >= 0 && changedProperties.has("sizeX")) { + this.element.style.width = `${this.element.sizeX}px`; + } + if (this.element.sizeY >= 0 && changedProperties.has("sizeY")) { + this.element.style.height = `${this.element.sizeY}px`; + } + } + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + this.element.append( + this.#THandler, + this.#RHandler, + this.#BHandler, + this.#LHandler, + this.#TRHandler, + this.#BRHandler, + this.#BLHandler, + this.#TLHandler + ); + } + + createInputObjects() { + return [ + ...super.createInputObjects(), + new MouseClickDrag(this.#THandler, this.blueprint, { + onDrag: (location, movement) => { + movement[1] = location[1] - this.element.topBoundary(); + if (this.setSizeY(this.element.sizeY - movement[1])) { + this.element.addLocation(0, movement[1], false); + } + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#RHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[0] = location[0] - this.element.rightBoundary(); + this.setSizeX(this.element.sizeX + movement[0]); + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#BHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[1] = location[1] - this.element.bottomBoundary(); + this.setSizeY(this.element.sizeY + movement[1]); + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#LHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[0] = location[0] - this.element.leftBoundary(); + if (this.setSizeX(this.element.sizeX - movement[0])) { + this.element.addLocation(movement[0], 0, false); + } + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#TRHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[0] = location[0] - this.element.rightBoundary(); + movement[1] = location[1] - this.element.topBoundary(); + this.setSizeX(this.element.sizeX + movement[0]); + if (this.setSizeY(this.element.sizeY - movement[1])) { + this.element.addLocation(0, movement[1], false); + } + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#BRHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[0] = location[0] - this.element.rightBoundary(); + movement[1] = location[1] - this.element.bottomBoundary(); + this.setSizeX(this.element.sizeX + movement[0]); + this.setSizeY(this.element.sizeY + movement[1]); + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#BLHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[0] = location[0] - this.element.leftBoundary(); + movement[1] = location[1] - this.element.bottomBoundary(); + if (this.setSizeX(this.element.sizeX - movement[0])) { + this.element.addLocation(movement[0], 0, false); + } + this.setSizeY(this.element.sizeY + movement[1]); + }, + onEndDrag: () => this.endResize(), + }), + new MouseClickDrag(this.#TLHandler, this.blueprint, { + onDrag: (location, movement) => { + movement[0] = location[0] - this.element.leftBoundary(); + movement[1] = location[1] - this.element.topBoundary(); + if (this.setSizeX(this.element.sizeX - movement[0])) { + this.element.addLocation(movement[0], 0, false); + } + if (this.setSizeY(this.element.sizeY - movement[1])) { + this.element.addLocation(0, movement[1], false); + } + }, + onEndDrag: () => this.endResize(), + }), + ] + } + + /** @param {Number} value */ + setSizeX(value) { + this.element.setNodeWidth(value); + return true + } + + /** @param {Number} value */ + setSizeY(value) { + this.element.setNodeHeight(value); + return true + } + + endResize() { + } +} + +/** + * @typedef {import("../../element/NodeElement").default} NodeElement + * @typedef {import("../../element/PinElement").default} PinElement + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +class CommentNodeTemplate extends IResizeableTemplate { + + #color = LinearColorEntity.getWhite() + #selectableAreaHeight = 0 + + /** @param {NodeElement} element */ + initialize(element) { + if (element.entity.CommentColor) { + this.#color.setFromRGBANumber(element.entity.CommentColor.toNumber()); + this.#color.setFromHSVA( + this.#color.H.value, + this.#color.S.value, + Math.pow(this.#color.V.value, 0.45) * 0.67 + ); + } + element.classList.add("ueb-node-style-comment", "ueb-node-resizeable"); + element.sizeX = 25 * Configuration.gridSize; + element.sizeY = 6 * Configuration.gridSize; + super.initialize(element); // Keep it at the end because it calls this.getColor() where this.#color must be initialized + } + + getColor() { + return i$3`${Math.round(this.#color.R.value * 255)}, ${Math.round(this.#color.G.value * 255)}, ${Math.round(this.#color.B.value * 255)}` + } + + getDraggableElement() { + return this.element.querySelector(".ueb-node-top") + } + + render() { + return y` +
+
+
+ ${this.element.entity.NodeComment} +
+
+
+ ` + } + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + const bounding = this.getDraggableElement().getBoundingClientRect(); + this.#selectableAreaHeight = bounding.height; + } + + manageNodesBind() { + let nodes = this.blueprint.getNodes(); + for (let node of nodes) { + if ( + node.topBoundary() >= this.element.topBoundary() + && node.rightBoundary() <= this.element.rightBoundary() + && node.bottomBoundary() <= this.element.bottomBoundary() + && node.leftBoundary() >= this.element.leftBoundary() + ) { + node.bindToComment(this.element); + } else { + node.unbindFromComment(this.element); + } + } + } + + /** @param {Number} value */ + setSizeX(value) { + value = Math.round(value); + if (value >= Configuration.gridSet * Configuration.gridSize) { + this.element.setNodeWidth(value); + return true + } + return false + } + + /** @param {Number} value */ + setSizeY(value) { + value = Math.round(value); + if (value >= 3 * Configuration.gridSize) { + this.element.setNodeHeight(value); + return true + } + return false + } + + endResize() { + this.manageNodesBind(); + } + + topBoundary(justSelectableArea = false) { + return this.element.locationY + } + + rightBoundary(justSelectableArea = false) { + return this.element.locationX + this.element.sizeX + } + + bottomBoundary(justSelectableArea = false) { + return justSelectableArea + ? this.element.locationY + this.#selectableAreaHeight + : super.bottomBoundary() + } + + leftBoundary(justSelectableArea = false) { + return this.element.locationX + } +} + /** * @typedef {import("../../Blueprint").default} Blueprint * @typedef {import("../../element/LinkElement").default} LinkElement @@ -6644,80 +6684,80 @@ class MouseCreateLink extends IMouseClickDrag { this.link = null; this.#listenedPins = null; } -} - -/** - * @typedef {import("../../element/NodeElement").default} NodeElement - * @typedef {import("../../element/PinElement").PinElementConstructor} PinElementConstructor - */ - -class VariableManagementNodeTemplate extends NodeTemplate { - - #hasInput = false - #hasOutput = false - #displayName = "" - - static nodeStyleClasses = ["ueb-node-style-glass"] - - /** @param {NodeElement} element */ - initialize(element) { - super.initialize(element); - this.#displayName = this.element.getNodeDisplayName(); - } - - render() { - return y` -
-
- ${this.#displayName ? y` -
-
- - ${this.#displayName} - -
-
- ` : b} -
- ${this.#hasInput ? y` -
- ` : b} - ${this.#hasOutput ? y` -
- ` : b} -
-
-
- ` - } - - createPinElements() { - return this.element.getPinEntities() - .filter(v => !v.isHidden()) - .map(v => { - this.#hasInput ||= v.isInput(); - this.#hasOutput ||= v.isOutput(); - const result = /** @type {PinElementConstructor} */(ElementFactory.getConstructor("ueb-pin")) - .newObject(v, undefined, this.element); - return result - }) - } -} - -/** @typedef {import("../../element/NodeElement").default} NodeElement */ - -class VariableConversionNodeTemplate extends VariableManagementNodeTemplate { - - static nodeStyleClasses = [...super.nodeStyleClasses, "ueb-node-style-conversion"] -} - -/** @typedef {import("../../element/NodeElement").default} NodeElement */ - -class VariableOperationNodeTemplate extends VariableManagementNodeTemplate { - - static nodeStyleClasses = [...super.nodeStyleClasses, "ueb-node-style-operation"] -} - +} + +/** + * @typedef {import("../../element/NodeElement").default} NodeElement + * @typedef {import("../../element/PinElement").PinElementConstructor} PinElementConstructor + */ + +class VariableManagementNodeTemplate extends NodeTemplate { + + #hasInput = false + #hasOutput = false + #displayName = "" + + static nodeStyleClasses = ["ueb-node-style-glass"] + + /** @param {NodeElement} element */ + initialize(element) { + super.initialize(element); + this.#displayName = this.element.getNodeDisplayName(); + } + + render() { + return y` +
+
+ ${this.#displayName ? y` +
+
+ + ${this.#displayName} + +
+
+ ` : b} +
+ ${this.#hasInput ? y` +
+ ` : b} + ${this.#hasOutput ? y` +
+ ` : b} +
+
+
+ ` + } + + createPinElements() { + return this.element.getPinEntities() + .filter(v => !v.isHidden()) + .map(v => { + this.#hasInput ||= v.isInput(); + this.#hasOutput ||= v.isOutput(); + const result = /** @type {PinElementConstructor} */(ElementFactory.getConstructor("ueb-pin")) + .newObject(v, undefined, this.element); + return result + }) + } +} + +/** @typedef {import("../../element/NodeElement").default} NodeElement */ + +class VariableConversionNodeTemplate extends VariableManagementNodeTemplate { + + static nodeStyleClasses = [...super.nodeStyleClasses, "ueb-node-style-conversion"] +} + +/** @typedef {import("../../element/NodeElement").default} NodeElement */ + +class VariableOperationNodeTemplate extends VariableManagementNodeTemplate { + + static nodeStyleClasses = [...super.nodeStyleClasses, "ueb-node-style-operation"] +} + /** * @typedef {import("../../input/IInput").default} IInput * @typedef {import("lit").PropertyValues} PropertyValues @@ -6838,28 +6878,28 @@ class PinTemplate extends ITemplate { getClickableElement() { return this.#wrapperElement ?? this.element } -} - -/** - * @template T - * @typedef {import("../../element/PinElement").default} PinElement - */ - -/** - * @template T - * @extends PinTemplate> - */ -class MinimalPinTemplate extends PinTemplate { - - render() { - return y` -
-
${this.renderIcon()}
-
- ` - } -} - +} + +/** + * @template T + * @typedef {import("../../element/PinElement").default} PinElement + */ + +/** + * @template T + * @extends PinTemplate> + */ +class MinimalPinTemplate extends PinTemplate { + + render() { + return y` +
+
${this.renderIcon()}
+
+ ` + } +} + /** * @typedef {import("../../element/PinElement").PinElementConstructor} PinElementConstructor * @typedef {import("lit").PropertyValues} PropertyValues @@ -6914,492 +6954,492 @@ class EventNodeTemplate extends NodeTemplate { .newObject(pinEntity, undefined, this.element) ) } -} - -/** - * @typedef {import("../element/IDraggableElement").DragEvent} DragEvent - * @typedef {import("../entity/IEntity").default} IEntity - * @typedef {import("../template/ISelectableDraggableTemplate").default} ISelectableDraggableTemplate - */ - -/** - * @template {IEntity} T - * @template {ISelectableDraggableTemplate} U - * @extends {IDraggableElement} - */ -class ISelectableDraggableElement extends IDraggableElement { - - static properties = { - ...super.properties, - selected: { - type: Boolean, - attribute: "data-selected", - reflect: true, - converter: Utility.booleanConverter, - }, - } - - /** @param {DragEvent} e */ - dragHandler = e => this.addLocation(...e.detail.value) - - constructor() { - super(); - this.selected = false; - this.listeningDrag = false; - } - - setup() { - super.setup(); - this.setSelected(this.selected); - } - - cleanup() { - super.cleanup(); - this.blueprint.removeEventListener(Configuration.nodeDragGeneralEventName, this.dragHandler); - } - - setSelected(value = true) { - this.selected = value; - if (this.blueprint) { - if (this.selected) { - this.listeningDrag = true; - this.blueprint.addEventListener(Configuration.nodeDragGeneralEventName, this.dragHandler); - } else { - this.blueprint.removeEventListener(Configuration.nodeDragGeneralEventName, this.dragHandler); - this.listeningDrag = false; - } - } - } -} - -/** - * @typedef {import("../node/KnotNodeTemplate").default} KnotNodeTemplate - * @typedef {import("../../entity/PinEntity").default} KnotEntity - */ - -/** @extends MinimalPinTemplate */ -class KnotPinTemplate extends MinimalPinTemplate { - - render() { - return this.element.isOutput() ? super.render() : y`` - } - - getOppositePin() { - const nodeTemplate = /** @type {KnotNodeTemplate} */(this.element.nodeElement.template); - return this.element.isOutput() ? nodeTemplate.inputPin : nodeTemplate.outputPin - } - - getLinkLocation() { - const rect = ( - this.element.isInput() - ? /** @type {KnotNodeTemplate} */(this.element.nodeElement.template).outputPin.template - : this - ) - .iconElement.getBoundingClientRect(); - const boundingLocation = [this.element.isInput() ? rect.left : rect.right, (rect.top + rect.bottom) / 2]; - const location = Utility.convertLocation(boundingLocation, this.blueprint.template.gridElement); - return this.blueprint.compensateTranslation(location[0], location[1]) - } -} - -/** - * @typedef {import("../../element/NodeElement").default} NodeElement - * @typedef {import("../../element/PinElement").default} PinElement - * @typedef {import("../../element/PinElement").PinElementConstructor} PinElementConstructor - */ - -class KnotNodeTemplate extends NodeTemplate { - - static #traversedPin = new Set() - - /** @type {Boolean?} */ - #chainDirection = null // The node is part of a chain connected to an input or output pin - - /** @type {PinElement} */ - #inputPin - get inputPin() { - return this.#inputPin - } - - /** @type {PinElement} */ - #outputPin - get outputPin() { - return this.#outputPin - } - - /** @param {NodeElement} element */ - initialize(element) { - super.initialize(element); - this.element.classList.add("ueb-node-style-minimal"); - } - - /** @param {PinElement} startingPin */ - findDirectionaPin(startingPin) { - if ( - startingPin.nodeElement.getType() !== Configuration.nodeType.knot - || KnotNodeTemplate.#traversedPin.has(startingPin) - ) { - KnotNodeTemplate.#traversedPin.clear(); - return true - } - KnotNodeTemplate.#traversedPin.add(startingPin); - for (let pin of startingPin.getLinks().map(l => this.blueprint.getPin(l))) { - if (this.findDirectionaPin(pin)) { - return true - } - } - return false - } - - render() { - return y` -
- ` - } - - setupPins() { - this.element.getPinElements().forEach( - p => /** @type {HTMLElement} */(this.element.querySelector(".ueb-node-border")).appendChild(p) - ); - } - - /** - * @param {NodeElement} node - * @returns {NodeListOf} - */ - getPinElements(node) { - return node.querySelectorAll("ueb-pin") - } - - createPinElements() { - const entities = this.element.getPinEntities().filter(v => !v.isHidden()); - const inputEntity = entities[entities[0].isInput() ? 0 : 1]; - const outputEntity = entities[entities[0].isOutput() ? 0 : 1]; - const pinElementConstructor = /** @type {PinElementConstructor} */(ElementFactory.getConstructor("ueb-pin")); - let result = [ - this.#inputPin = pinElementConstructor.newObject(inputEntity, new KnotPinTemplate(), this.element), - this.#outputPin = pinElementConstructor.newObject(outputEntity, new KnotPinTemplate(), this.element), - ]; - return result - } - - linksChanged() { - - } -} - -/** @typedef {import("../../element/NodeElement").default} NodeElement */ - -class VariableAccessNodeTemplate extends VariableManagementNodeTemplate { - - /** @param {NodeElement} element */ - initialize(element) { - super.initialize(element); - if (element.getType() === Configuration.nodeType.variableGet) { - this.element.classList.add("ueb-node-style-getter"); - } else if (element.getType() === Configuration.nodeType.variableSet) { - this.element.classList.add("ueb-node-style-setter"); - } - } - - setupPins() { - super.setupPins(); - let outputPin = this.element.getPinElements().find(p => !p.entity.isHidden() && !p.entity.isExecution()); - this.element.style.setProperty("--ueb-node-color", outputPin.getColor().cssText); - } -} - -/** - * @typedef {import("./IDraggableElement").DragEvent} DragEvent - * @typedef {import("./IElement").default} IElement - * @typedef {import("./PinElement").default} PinElement - * @typedef {typeof NodeElement} NodeElementConstructor - */ - -/** @extends {ISelectableDraggableElement} */ -class NodeElement extends ISelectableDraggableElement { - - static properties = { - ...ISelectableDraggableElement.properties, - typePath: { - type: String, - attribute: "data-type", - reflect: true, - }, - nodeName: { - type: String, - attribute: "data-name", - reflect: true, - }, - advancedPinDisplay: { - type: String, - attribute: "data-advanced-display", - converter: IdentifierEntity.attributeConverter, - reflect: true, - }, - enabledState: { - type: String, - attribute: "data-enabled-state", - reflect: true, - }, - nodeDisplayName: { - type: String, - attribute: false, - }, - pureFunction: { - type: Boolean, - converter: Utility.booleanConverter, - attribute: "data-pure-function", - reflect: true, - }, - } - static dragEventName = Configuration.nodeDragEventName - static dragGeneralEventName = Configuration.nodeDragGeneralEventName - - get blueprint() { - return super.blueprint - } - set blueprint(v) { - super.blueprint = v; - this.#pins.forEach(p => p.blueprint = v); - } - - /** @type {HTMLElement} */ - #nodeNameElement - get nodeNameElement() { - return this.#nodeNameElement - } - set nodeNameElement(value) { - this.#nodeNameElement = value; - } - - /** @type {PinElement[]} */ - #pins = [] - /** @type {NodeElement[]} */ - boundComments = [] - #commentDragged = false - /** @param {DragEvent} e */ - #commentDragHandler = e => { - // If selected, it will already drag, also must check if under nested comments, it must drag just once - if (!this.selected && !this.#commentDragged) { - this.#commentDragged = true; - this.addNextUpdatedCallbacks(() => this.#commentDragged = false); - this.addLocation(...e.detail.value); - } - } - - /** - * @param {ObjectEntity} nodeEntity - * @return {new () => NodeTemplate} - */ - static getTypeTemplate(nodeEntity) { - if ( - nodeEntity.getClass() === Configuration.nodeType.callFunction - || nodeEntity.getClass() === Configuration.nodeType.commutativeAssociativeBinaryOperator - || nodeEntity.getClass() === Configuration.nodeType.callArrayFunction - ) { - const memberParent = nodeEntity.FunctionReference.MemberParent?.path ?? ""; - if ( - memberParent === "/Script/Engine.KismetMathLibrary" - || memberParent === "/Script/Engine.KismetArrayLibrary" - ) { - if (nodeEntity.FunctionReference.MemberName?.startsWith("Conv_")) { - return VariableConversionNodeTemplate - } - if (nodeEntity.FunctionReference.MemberName?.startsWith("Percent_")) { - return VariableOperationNodeTemplate - } - switch (nodeEntity.FunctionReference.MemberName) { - case "Array_Add": - case "Array_Identical": - case "Abs": - case "Array_Add": - case "BMax": - case "BMin": - case "Exp": - case "FMax": - case "FMin": - case "Max": - case "MaxInt64": - case "Min": - case "MinInt64": - return VariableOperationNodeTemplate - } - } - if (memberParent === "/Script/Engine.BlueprintSetLibrary") { - return VariableOperationNodeTemplate - } - if (memberParent === "/Script/Engine.BlueprintMapLibrary") { - return VariableOperationNodeTemplate - } - } - switch (nodeEntity.getClass()) { - case Configuration.nodeType.comment: - return CommentNodeTemplate - case Configuration.nodeType.event: - case Configuration.nodeType.customEvent: - return EventNodeTemplate - case Configuration.nodeType.promotableOperator: - return VariableOperationNodeTemplate - case Configuration.nodeType.knot: return KnotNodeTemplate - case Configuration.nodeType.variableGet: return VariableAccessNodeTemplate - case Configuration.nodeType.variableSet: return VariableAccessNodeTemplate - } - if (nodeEntity.getDelegatePin()) { - return EventNodeTemplate - } - return NodeTemplate - } - - /** @param {String} str */ - static fromSerializedObject(str) { - str = str.trim(); - let entity = SerializerFactory.getSerializer(ObjectEntity).deserialize(str); - return NodeElement.newObject(/** @type {ObjectEntity} */(entity)) - } - - /** - * @param {ObjectEntity} entity - * @param {NodeTemplate} template - */ - static newObject(entity = new ObjectEntity(), template = new (NodeElement.getTypeTemplate(entity))()) { - const result = new NodeElement(); - result.initialize(entity, template); - return result - } - - initialize(entity = new ObjectEntity(), template = new (NodeElement.getTypeTemplate(entity))()) { - super.initialize(entity, template); - this.#pins = this.template.createPinElements(); - this.typePath = this.entity.getType(); - this.nodeName = this.entity.getObjectName(); - this.advancedPinDisplay = this.entity.AdvancedPinDisplay?.toString(); - this.enabledState = this.entity.EnabledState; - this.nodeDisplayName = this.getNodeDisplayName(); - this.pureFunction = this.entity.bIsPureFunc; - this.dragLinkObjects = []; - super.setLocation(this.entity.getNodePosX(), this.entity.getNodePosY()); - if (this.entity.NodeWidth && this.entity.NodeHeight) { - this.sizeX = this.entity.NodeWidth.value; - this.sizeY = this.entity.NodeHeight.value; - } else { - this.updateComplete.then(() => this.computeSizes()); - } - } - - getUpdateComplete() { - return Promise.all([ - super.getUpdateComplete(), - ...this.getPinElements().map(pin => pin.updateComplete) - ]).then(() => true) - } - - /** @param {NodeElement} commentNode */ - bindToComment(commentNode) { - if (commentNode != this && !this.boundComments.includes(commentNode)) { - commentNode.addEventListener(Configuration.nodeDragEventName, this.#commentDragHandler); - this.boundComments.push(commentNode); - } - } - - /** @param {NodeElement} commentNode */ - unbindFromComment(commentNode) { - const commentIndex = this.boundComments.indexOf(commentNode); - if (commentIndex >= 0) { - commentNode.removeEventListener(Configuration.nodeDragEventName, this.#commentDragHandler); - this.boundComments[commentIndex] = this.boundComments[this.boundComments.length - 1]; - this.boundComments.pop(); - } - } - - /** @param {NodeElement} commentNode */ - isInsideComment(commentNode) { - return this.topBoundary() >= commentNode.topBoundary() - && this.rightBoundary() <= commentNode.rightBoundary() - && this.bottomBoundary() <= commentNode.bottomBoundary() - && this.leftBoundary() >= commentNode.leftBoundary() - } - - getType() { - return this.entity.getType() - } - - getNodeName() { - return this.entity.getObjectName() - } - - getNodeDisplayName() { - return this.entity.nodeDisplayName() - } - - /** @param {Number} value */ - setNodeWidth(value) { - this.entity.setNodeWidth(value); - this.sizeX = value; - this.acknowledgeReflow(); - } - - /** @param {Number} value */ - setNodeHeight(value) { - this.entity.setNodeHeight(value); - this.sizeY = value; - this.acknowledgeReflow(); - } - - /** @param {IElement[]} nodesWhitelist */ - sanitizeLinks(nodesWhitelist = []) { - this.getPinElements().forEach(pin => pin.sanitizeLinks(nodesWhitelist)); - } - - /** @param {String} name */ - rename(name) { - if (this.entity.Name == name) { - return false - } - for (let sourcePinElement of this.getPinElements()) { - for (let targetPinReference of sourcePinElement.getLinks()) { - this.blueprint.getPin(targetPinReference).redirectLink(sourcePinElement, new PinReferenceEntity({ - objectName: name, - pinGuid: sourcePinElement.entity.PinId, - })); - } - } - this.entity.Name = name; - this.nodeName = this.entity.Name; - } - - getPinElements() { - return this.#pins - } - - /** @returns {PinEntity[]} */ - getPinEntities() { - return this.entity.CustomProperties.filter(v => v instanceof PinEntity) - } - - setLocation(x = 0, y = 0, acknowledge = true) { - this.entity.setNodePosX(x); - this.entity.setNodePosY(y); - super.setLocation(x, y, acknowledge); - } - - acknowledgeReflow() { - this.requestUpdate(); - this.updateComplete.then(() => this.computeSizes()); - let reflowEvent = new CustomEvent(Configuration.nodeReflowEventName); - this.dispatchEvent(reflowEvent); - } - - setShowAdvancedPinDisplay(value) { - this.entity.AdvancedPinDisplay = new IdentifierEntity(value ? "Shown" : "Hidden"); - this.advancedPinDisplay = this.entity.AdvancedPinDisplay; - } - - toggleShowAdvancedPinDisplay() { - this.setShowAdvancedPinDisplay(this.entity.AdvancedPinDisplay?.toString() != "Shown"); - } -} - +} + +/** + * @typedef {import("../element/IDraggableElement").DragEvent} DragEvent + * @typedef {import("../entity/IEntity").default} IEntity + * @typedef {import("../template/ISelectableDraggableTemplate").default} ISelectableDraggableTemplate + */ + +/** + * @template {IEntity} T + * @template {ISelectableDraggableTemplate} U + * @extends {IDraggableElement} + */ +class ISelectableDraggableElement extends IDraggableElement { + + static properties = { + ...super.properties, + selected: { + type: Boolean, + attribute: "data-selected", + reflect: true, + converter: Utility.booleanConverter, + }, + } + + /** @param {DragEvent} e */ + dragHandler = e => this.addLocation(...e.detail.value) + + constructor() { + super(); + this.selected = false; + this.listeningDrag = false; + } + + setup() { + super.setup(); + this.setSelected(this.selected); + } + + cleanup() { + super.cleanup(); + this.blueprint.removeEventListener(Configuration.nodeDragGeneralEventName, this.dragHandler); + } + + setSelected(value = true) { + this.selected = value; + if (this.blueprint) { + if (this.selected) { + this.listeningDrag = true; + this.blueprint.addEventListener(Configuration.nodeDragGeneralEventName, this.dragHandler); + } else { + this.blueprint.removeEventListener(Configuration.nodeDragGeneralEventName, this.dragHandler); + this.listeningDrag = false; + } + } + } +} + +/** + * @typedef {import("../node/KnotNodeTemplate").default} KnotNodeTemplate + * @typedef {import("../../entity/PinEntity").default} KnotEntity + */ + +/** @extends MinimalPinTemplate */ +class KnotPinTemplate extends MinimalPinTemplate { + + render() { + return this.element.isOutput() ? super.render() : y`` + } + + getOppositePin() { + const nodeTemplate = /** @type {KnotNodeTemplate} */(this.element.nodeElement.template); + return this.element.isOutput() ? nodeTemplate.inputPin : nodeTemplate.outputPin + } + + getLinkLocation() { + const rect = ( + this.element.isInput() + ? /** @type {KnotNodeTemplate} */(this.element.nodeElement.template).outputPin.template + : this + ) + .iconElement.getBoundingClientRect(); + const boundingLocation = [this.element.isInput() ? rect.left : rect.right, (rect.top + rect.bottom) / 2]; + const location = Utility.convertLocation(boundingLocation, this.blueprint.template.gridElement); + return this.blueprint.compensateTranslation(location[0], location[1]) + } +} + +/** + * @typedef {import("../../element/NodeElement").default} NodeElement + * @typedef {import("../../element/PinElement").default} PinElement + * @typedef {import("../../element/PinElement").PinElementConstructor} PinElementConstructor + */ + +class KnotNodeTemplate extends NodeTemplate { + + static #traversedPin = new Set() + + /** @type {Boolean?} */ + #chainDirection = null // The node is part of a chain connected to an input or output pin + + /** @type {PinElement} */ + #inputPin + get inputPin() { + return this.#inputPin + } + + /** @type {PinElement} */ + #outputPin + get outputPin() { + return this.#outputPin + } + + /** @param {NodeElement} element */ + initialize(element) { + super.initialize(element); + this.element.classList.add("ueb-node-style-minimal"); + } + + /** @param {PinElement} startingPin */ + findDirectionaPin(startingPin) { + if ( + startingPin.nodeElement.getType() !== Configuration.nodeType.knot + || KnotNodeTemplate.#traversedPin.has(startingPin) + ) { + KnotNodeTemplate.#traversedPin.clear(); + return true + } + KnotNodeTemplate.#traversedPin.add(startingPin); + for (let pin of startingPin.getLinks().map(l => this.blueprint.getPin(l))) { + if (this.findDirectionaPin(pin)) { + return true + } + } + return false + } + + render() { + return y` +
+ ` + } + + setupPins() { + this.element.getPinElements().forEach( + p => /** @type {HTMLElement} */(this.element.querySelector(".ueb-node-border")).appendChild(p) + ); + } + + /** + * @param {NodeElement} node + * @returns {NodeListOf} + */ + getPinElements(node) { + return node.querySelectorAll("ueb-pin") + } + + createPinElements() { + const entities = this.element.getPinEntities().filter(v => !v.isHidden()); + const inputEntity = entities[entities[0].isInput() ? 0 : 1]; + const outputEntity = entities[entities[0].isOutput() ? 0 : 1]; + const pinElementConstructor = /** @type {PinElementConstructor} */(ElementFactory.getConstructor("ueb-pin")); + let result = [ + this.#inputPin = pinElementConstructor.newObject(inputEntity, new KnotPinTemplate(), this.element), + this.#outputPin = pinElementConstructor.newObject(outputEntity, new KnotPinTemplate(), this.element), + ]; + return result + } + + linksChanged() { + + } +} + +/** @typedef {import("../../element/NodeElement").default} NodeElement */ + +class VariableAccessNodeTemplate extends VariableManagementNodeTemplate { + + /** @param {NodeElement} element */ + initialize(element) { + super.initialize(element); + if (element.getType() === Configuration.nodeType.variableGet) { + this.element.classList.add("ueb-node-style-getter"); + } else if (element.getType() === Configuration.nodeType.variableSet) { + this.element.classList.add("ueb-node-style-setter"); + } + } + + setupPins() { + super.setupPins(); + let outputPin = this.element.getPinElements().find(p => !p.entity.isHidden() && !p.entity.isExecution()); + this.element.style.setProperty("--ueb-node-color", outputPin.getColor().cssText); + } +} + +/** + * @typedef {import("./IDraggableElement").DragEvent} DragEvent + * @typedef {import("./IElement").default} IElement + * @typedef {import("./PinElement").default} PinElement + * @typedef {typeof NodeElement} NodeElementConstructor + */ + +/** @extends {ISelectableDraggableElement} */ +class NodeElement extends ISelectableDraggableElement { + + static properties = { + ...ISelectableDraggableElement.properties, + typePath: { + type: String, + attribute: "data-type", + reflect: true, + }, + nodeName: { + type: String, + attribute: "data-name", + reflect: true, + }, + advancedPinDisplay: { + type: String, + attribute: "data-advanced-display", + converter: IdentifierEntity.attributeConverter, + reflect: true, + }, + enabledState: { + type: String, + attribute: "data-enabled-state", + reflect: true, + }, + nodeDisplayName: { + type: String, + attribute: false, + }, + pureFunction: { + type: Boolean, + converter: Utility.booleanConverter, + attribute: "data-pure-function", + reflect: true, + }, + } + static dragEventName = Configuration.nodeDragEventName + static dragGeneralEventName = Configuration.nodeDragGeneralEventName + + get blueprint() { + return super.blueprint + } + set blueprint(v) { + super.blueprint = v; + this.#pins.forEach(p => p.blueprint = v); + } + + /** @type {HTMLElement} */ + #nodeNameElement + get nodeNameElement() { + return this.#nodeNameElement + } + set nodeNameElement(value) { + this.#nodeNameElement = value; + } + + /** @type {PinElement[]} */ + #pins = [] + /** @type {NodeElement[]} */ + boundComments = [] + #commentDragged = false + /** @param {DragEvent} e */ + #commentDragHandler = e => { + // If selected, it will already drag, also must check if under nested comments, it must drag just once + if (!this.selected && !this.#commentDragged) { + this.#commentDragged = true; + this.addNextUpdatedCallbacks(() => this.#commentDragged = false); + this.addLocation(...e.detail.value); + } + } + + /** + * @param {ObjectEntity} nodeEntity + * @return {new () => NodeTemplate} + */ + static getTypeTemplate(nodeEntity) { + if ( + nodeEntity.getClass() === Configuration.nodeType.callFunction + || nodeEntity.getClass() === Configuration.nodeType.commutativeAssociativeBinaryOperator + || nodeEntity.getClass() === Configuration.nodeType.callArrayFunction + ) { + const memberParent = nodeEntity.FunctionReference.MemberParent?.path ?? ""; + if ( + memberParent === "/Script/Engine.KismetMathLibrary" + || memberParent === "/Script/Engine.KismetArrayLibrary" + ) { + if (nodeEntity.FunctionReference.MemberName?.startsWith("Conv_")) { + return VariableConversionNodeTemplate + } + if (nodeEntity.FunctionReference.MemberName?.startsWith("Percent_")) { + return VariableOperationNodeTemplate + } + switch (nodeEntity.FunctionReference.MemberName) { + case "Array_Add": + case "Array_Identical": + case "Abs": + case "Array_Add": + case "BMax": + case "BMin": + case "Exp": + case "FMax": + case "FMin": + case "Max": + case "MaxInt64": + case "Min": + case "MinInt64": + return VariableOperationNodeTemplate + } + } + if (memberParent === "/Script/Engine.BlueprintSetLibrary") { + return VariableOperationNodeTemplate + } + if (memberParent === "/Script/Engine.BlueprintMapLibrary") { + return VariableOperationNodeTemplate + } + } + switch (nodeEntity.getClass()) { + case Configuration.nodeType.comment: + return CommentNodeTemplate + case Configuration.nodeType.event: + case Configuration.nodeType.customEvent: + return EventNodeTemplate + case Configuration.nodeType.promotableOperator: + return VariableOperationNodeTemplate + case Configuration.nodeType.knot: return KnotNodeTemplate + case Configuration.nodeType.variableGet: return VariableAccessNodeTemplate + case Configuration.nodeType.variableSet: return VariableAccessNodeTemplate + } + if (nodeEntity.getDelegatePin()) { + return EventNodeTemplate + } + return NodeTemplate + } + + /** @param {String} str */ + static fromSerializedObject(str) { + str = str.trim(); + let entity = SerializerFactory.getSerializer(ObjectEntity).deserialize(str); + return NodeElement.newObject(/** @type {ObjectEntity} */(entity)) + } + + /** + * @param {ObjectEntity} entity + * @param {NodeTemplate} template + */ + static newObject(entity = new ObjectEntity(), template = new (NodeElement.getTypeTemplate(entity))()) { + const result = new NodeElement(); + result.initialize(entity, template); + return result + } + + initialize(entity = new ObjectEntity(), template = new (NodeElement.getTypeTemplate(entity))()) { + super.initialize(entity, template); + this.#pins = this.template.createPinElements(); + this.typePath = this.entity.getType(); + this.nodeName = this.entity.getObjectName(); + this.advancedPinDisplay = this.entity.AdvancedPinDisplay?.toString(); + this.enabledState = this.entity.EnabledState; + this.nodeDisplayName = this.getNodeDisplayName(); + this.pureFunction = this.entity.bIsPureFunc; + this.dragLinkObjects = []; + super.setLocation(this.entity.getNodePosX(), this.entity.getNodePosY()); + if (this.entity.NodeWidth && this.entity.NodeHeight) { + this.sizeX = this.entity.NodeWidth.value; + this.sizeY = this.entity.NodeHeight.value; + } else { + this.updateComplete.then(() => this.computeSizes()); + } + } + + getUpdateComplete() { + return Promise.all([ + super.getUpdateComplete(), + ...this.getPinElements().map(pin => pin.updateComplete) + ]).then(() => true) + } + + /** @param {NodeElement} commentNode */ + bindToComment(commentNode) { + if (commentNode != this && !this.boundComments.includes(commentNode)) { + commentNode.addEventListener(Configuration.nodeDragEventName, this.#commentDragHandler); + this.boundComments.push(commentNode); + } + } + + /** @param {NodeElement} commentNode */ + unbindFromComment(commentNode) { + const commentIndex = this.boundComments.indexOf(commentNode); + if (commentIndex >= 0) { + commentNode.removeEventListener(Configuration.nodeDragEventName, this.#commentDragHandler); + this.boundComments[commentIndex] = this.boundComments[this.boundComments.length - 1]; + this.boundComments.pop(); + } + } + + /** @param {NodeElement} commentNode */ + isInsideComment(commentNode) { + return this.topBoundary() >= commentNode.topBoundary() + && this.rightBoundary() <= commentNode.rightBoundary() + && this.bottomBoundary() <= commentNode.bottomBoundary() + && this.leftBoundary() >= commentNode.leftBoundary() + } + + getType() { + return this.entity.getType() + } + + getNodeName() { + return this.entity.getObjectName() + } + + getNodeDisplayName() { + return this.entity.nodeDisplayName() + } + + /** @param {Number} value */ + setNodeWidth(value) { + this.entity.setNodeWidth(value); + this.sizeX = value; + this.acknowledgeReflow(); + } + + /** @param {Number} value */ + setNodeHeight(value) { + this.entity.setNodeHeight(value); + this.sizeY = value; + this.acknowledgeReflow(); + } + + /** @param {IElement[]} nodesWhitelist */ + sanitizeLinks(nodesWhitelist = []) { + this.getPinElements().forEach(pin => pin.sanitizeLinks(nodesWhitelist)); + } + + /** @param {String} name */ + rename(name) { + if (this.entity.Name == name) { + return false + } + for (let sourcePinElement of this.getPinElements()) { + for (let targetPinReference of sourcePinElement.getLinks()) { + this.blueprint.getPin(targetPinReference).redirectLink(sourcePinElement, new PinReferenceEntity({ + objectName: name, + pinGuid: sourcePinElement.entity.PinId, + })); + } + } + this.entity.Name = name; + this.nodeName = this.entity.Name; + } + + getPinElements() { + return this.#pins + } + + /** @returns {PinEntity[]} */ + getPinEntities() { + return this.entity.CustomProperties.filter(v => v instanceof PinEntity) + } + + setLocation(x = 0, y = 0, acknowledge = true) { + this.entity.setNodePosX(x); + this.entity.setNodePosY(y); + super.setLocation(x, y, acknowledge); + } + + acknowledgeReflow() { + this.requestUpdate(); + this.updateComplete.then(() => this.computeSizes()); + let reflowEvent = new CustomEvent(Configuration.nodeReflowEventName); + this.dispatchEvent(reflowEvent); + } + + setShowAdvancedPinDisplay(value) { + this.entity.AdvancedPinDisplay = new IdentifierEntity(value ? "Shown" : "Hidden"); + this.advancedPinDisplay = this.entity.AdvancedPinDisplay; + } + + toggleShowAdvancedPinDisplay() { + this.setShowAdvancedPinDisplay(this.entity.AdvancedPinDisplay?.toString() != "Shown"); + } +} + /** * @typedef {import("./element/PinElement").default} PinElement * @typedef {import("./entity/GuidEntity").default} GuidEntity @@ -7856,65 +7896,65 @@ class Blueprint extends IElement { } } -customElements.define("ueb-blueprint", Blueprint); - -/** - * @typedef {import("../element/IDraggableElement").default} IDraggableElement - * @typedef {import("lit").PropertyValues} PropertyValues - */ - -/** - * @template {IDraggableElement} T - * @extends {IDraggableTemplate} - */ -class IDraggableControlTemplate extends IDraggableTemplate { - - /** @type {(x: Number, y: Number) => void} */ - #locationChangeCallback - get locationChangeCallback() { - return this.#locationChangeCallback - } - set locationChangeCallback(callback) { - this.#locationChangeCallback = callback; - } - - movementSpace - movementSpaceSize = [0, 0] - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - this.movementSpace = this.element.parentElement; - } - - setup() { - super.setup(); - const bounding = this.movementSpace.getBoundingClientRect(); - this.movementSpaceSize = [bounding.width, bounding.height]; - } - - createDraggableObject() { - return new MouseMoveDraggable(this.element, this.blueprint, { - draggableElement: this.movementSpace, - ignoreTranslateCompensate: true, - moveEverywhere: true, - movementSpace: this.movementSpace, - repositionOnClick: true, - stepSize: 1, - }) - } - - /** - * @param {Number} x - * @param {Number} y - * @returns {[Number, Number]} - */ - adjustLocation(x, y) { - this.locationChangeCallback?.(x, y); - return [x, y] - } -} - +customElements.define("ueb-blueprint", Blueprint); + +/** + * @typedef {import("../element/IDraggableElement").default} IDraggableElement + * @typedef {import("lit").PropertyValues} PropertyValues + */ + +/** + * @template {IDraggableElement} T + * @extends {IDraggableTemplate} + */ +class IDraggableControlTemplate extends IDraggableTemplate { + + /** @type {(x: Number, y: Number) => void} */ + #locationChangeCallback + get locationChangeCallback() { + return this.#locationChangeCallback + } + set locationChangeCallback(callback) { + this.#locationChangeCallback = callback; + } + + movementSpace + movementSpaceSize = [0, 0] + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + this.movementSpace = this.element.parentElement; + } + + setup() { + super.setup(); + const bounding = this.movementSpace.getBoundingClientRect(); + this.movementSpaceSize = [bounding.width, bounding.height]; + } + + createDraggableObject() { + return new MouseMoveDraggable(this.element, this.blueprint, { + draggableElement: this.movementSpace, + ignoreTranslateCompensate: true, + moveEverywhere: true, + movementSpace: this.movementSpace, + repositionOnClick: true, + stepSize: 1, + }) + } + + /** + * @param {Number} x + * @param {Number} y + * @returns {[Number, Number]} + */ + adjustLocation(x, y) { + this.locationChangeCallback?.(x, y); + return [x, y] + } +} + /** @typedef {import("../element/ColorHandlerElement").default} ColorHandlerElement */ /** @extends {IDraggableControlTemplate} */ @@ -7936,55 +7976,55 @@ class ColorHandlerTemplate extends IDraggableControlTemplate { y = Math.round(-y + radius); return [x, y] } -} - -/** - * @typedef {import("../element/WindowElement").default} WindowElement - * @typedef {import("../entity/IEntity").default} IEntity - * @typedef {import("../template/IDraggableControlTemplate").default} IDraggableControlTemplate - */ - -/** - * @template {IEntity} T - * @template {IDraggableControlTemplate} U - * @extends {IDraggableElement} - */ -class IDraggableControlElement extends IDraggableElement { - - /** @type {WindowElement} */ - windowElement - - setup() { - super.setup(); - this.windowElement = this.closest("ueb-window"); - } - - /** - * @param {Number} x - * @param {Number} y - */ - setLocation(x, y) { - super.setLocation(...this.template.adjustLocation(x, y)); - } -} - -/** @extends {IDraggableControlElement} */ -class ColorHandlerElement extends IDraggableControlElement { - - constructor() { - super(); - super.initialize({}, new ColorHandlerTemplate()); - } - - static newObject() { - return new ColorHandlerElement() - } - - initialize() { - // Initialized in the constructor, this method does nothing - } -} - +} + +/** + * @typedef {import("../element/WindowElement").default} WindowElement + * @typedef {import("../entity/IEntity").default} IEntity + * @typedef {import("../template/IDraggableControlTemplate").default} IDraggableControlTemplate + */ + +/** + * @template {IEntity} T + * @template {IDraggableControlTemplate} U + * @extends {IDraggableElement} + */ +class IDraggableControlElement extends IDraggableElement { + + /** @type {WindowElement} */ + windowElement + + setup() { + super.setup(); + this.windowElement = this.closest("ueb-window"); + } + + /** + * @param {Number} x + * @param {Number} y + */ + setLocation(x, y) { + super.setLocation(...this.template.adjustLocation(x, y)); + } +} + +/** @extends {IDraggableControlElement} */ +class ColorHandlerElement extends IDraggableControlElement { + + constructor() { + super(); + super.initialize({}, new ColorHandlerTemplate()); + } + + static newObject() { + return new ColorHandlerElement() + } + + initialize() { + // Initialized in the constructor, this method does nothing + } +} + /** @typedef {import("../element/ColorHandlerElement").default} ColorHandlerElement */ /** @extends {IDraggableControlTemplate} */ @@ -8001,416 +8041,416 @@ class ColorSliderTemplate extends IDraggableControlTemplate { this.locationChangeCallback?.(x / this.movementSpaceSize[0], 1 - y / this.movementSpaceSize[1]); return [x, y] } -} - -/** @extends {IDraggableControlElement} */ -class ColorSliderElement extends IDraggableControlElement { - - constructor() { - super(); - super.initialize({}, new ColorSliderTemplate()); - } - - static newObject() { - return new ColorSliderElement() - } - - initialize() { - // Initialized in the constructor, this method does nothing - } -} - -/** @typedef {import ("../../element/InputElement").default} InputElement */ - -/** @extends {ITemplate} */ -class InputTemplate extends ITemplate { - - #focusHandler = () => { - this.blueprint.acknowledgeEditText(true); - if (this.element.selectOnFocus) { - getSelection().selectAllChildren(this.element); - } - } - - #focusoutHandler = () => { - this.blueprint.acknowledgeEditText(false); - getSelection().removeAllRanges(); // Deselect eventually selected text inside the input - } - - /** @param {InputEvent} e */ - #inputSingleLineHandler = e => - /** @type {HTMLElement} */(e.target).querySelectorAll("br").forEach(br => br.remove()) - - /** @param {KeyboardEvent} e */ - #onKeydownBlurOnEnterHandler = e => { - if (e.code == "Enter" && !e.shiftKey) { - /** @type {HTMLElement} */(e.target).blur(); - } - } - - /** @param {InputElement} element */ - initialize(element) { - super.initialize(element); - this.element.classList.add("ueb-pin-input-content"); - this.element.setAttribute("role", "textbox"); - this.element.contentEditable = "true"; - } - - setup() { - super.setup(); - this.element.addEventListener("focus", this.#focusHandler); - this.element.addEventListener("focusout", this.#focusoutHandler); - if (this.element.singleLine) { - this.element.addEventListener("input", this.#inputSingleLineHandler); - } - if (this.element.blurOnEnter) { - this.element.addEventListener("keydown", this.#onKeydownBlurOnEnterHandler); - } - } - - cleanup() { - super.cleanup(); - this.element.removeEventListener("focus", this.#focusHandler); - this.element.removeEventListener("focusout", this.#focusoutHandler); - this.element.removeEventListener("input", this.#inputSingleLineHandler); - this.element.removeEventListener("keydown", this.#onKeydownBlurOnEnterHandler); - } -} - -class InputElement extends IElement { - - static properties = { - ...super.properties, - singleLine: { - type: Boolean, - attribute: "data-single-line", - converter: Utility.booleanConverter, - reflect: true, - }, - selectOnFocus: { - type: Boolean, - attribute: "data-select-focus", - converter: Utility.booleanConverter, - reflect: true, - }, - blurOnEnter: { - type: Boolean, - attribute: "data-blur-enter", - converter: Utility.booleanConverter, - reflect: true, - }, - } - - constructor() { - super(); - this.singleLine = false; - this.selectOnFocus = true; - this.blurOnEnter = true; - super.initialize({}, new InputTemplate()); - } - - static newObject() { - return new InputElement() - } - - initialize() { - // Initialized in the constructor, this method does nothing - } -} - -/** - * @typedef {import("../../element/IDraggableElement").default} IDraggableElement - */ - -/** -* @template {IDraggableElement} T -* @extends {IMouseClickDrag} -*/ -class MouseIgnore extends IMouseClickDrag { - - constructor(target, blueprint, options = {}) { - options.consumeEvent = true; - super(target, blueprint, options); - } -} - -/** @typedef {import("lit").PropertyValues} PropertyValues */ - -/** @extends PinTemplate */ -class BoolPinTemplate extends PinTemplate { - - /** @type {HTMLInputElement?} */ - #input - - #onChangeHandler = _ => this.element.setDefaultValue(this.#input.checked) - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - this.#input = this.element.querySelector(".ueb-pin-input"); - } - - setup() { - super.setup(); - this.#input?.addEventListener("change", this.#onChangeHandler); - } - - cleanup() { - super.cleanup(); - this.#input?.removeEventListener("change", this.#onChangeHandler); - } - - createInputObjects() { - return [ - ...super.createInputObjects(), - new MouseIgnore(this.#input, this.blueprint), - ] - } - - renderInput() { - return y` - - ` - } -} - -/** @typedef {import("../../element/PinElement").default} PinElement */ - -class ExecPinTemplate extends PinTemplate { - - renderIcon() { - return SVGIcon.execPin - } - - renderName() { - let pinName = this.element.entity.PinName; - if (this.element.entity.PinFriendlyName) { - pinName = this.element.entity.PinFriendlyName.toString(); - } else if (pinName === "execute" || pinName === "then") { - return y`` - } - return y`${Utility.formatStringName(pinName)}` - } -} - -/** @typedef {import("lit").PropertyValues} PropertyValues */ - -/** - * @template T - * @extends PinTemplate - */ -class IInputPinTemplate extends PinTemplate { - - static singleLineInput = false - static selectOnFocus = true - - /** @type {HTMLElement[]} */ - #inputContentElements - get inputContentElements() { - return this.#inputContentElements - } - - /** @param {String} value */ - static stringFromInputToUE(value) { - return value - .replace(/(?=\n\s*)\n$/, "") // Remove trailing double newline - .replaceAll("\n", "\\r\n") // Replace newline with \r\n (default newline in UE) - } - - /** @param {String} value */ - static stringFromUEToInput(value) { - return value - .replaceAll(/(?:\r|(?<=(?:^|[^\\])(?:\\\\)*)\\r)(?=\n)/g, "") // Remove \r leftover from \r\n - .replace(/(?<=\n\s*)$/, "\n") // Put back trailing double newline - } - - #onFocusOutHandler = () => this.setInputs(this.getInputs(), true) - /** @param {InputEvent} event */ - #onInputCheckWrapHandler = event => this.#updateWrapClass(/** @type {HTMLElement} */(event.target)) - - /** @param {HTMLElement} inputElement*/ - #updateWrapClass(inputElement) { - const width = this.blueprint.scaleCorrect(inputElement.getBoundingClientRect().width) + this.nameWidth; - const inputWrapped = this.element.classList.contains("ueb-pin-input-wrap"); - if (!inputWrapped && width > Configuration.pinInputWrapWidth) { - this.element.classList.add("ueb-pin-input-wrap"); - } else if (inputWrapped && width <= Configuration.pinInputWrapWidth) { - this.element.classList.remove("ueb-pin-input-wrap"); - } - } - - /** @param {PropertyValues} changedProperties */ - firstUpdated(changedProperties) { - super.firstUpdated(changedProperties); - this.#inputContentElements = /** @type {HTMLElement[]} */([...this.element.querySelectorAll("ueb-input")]); - if (/** @type {typeof IInputPinTemplate} */(this.constructor).canWrapInput) { - this.nameWidth = this.blueprint.scaleCorrect( - this.element.querySelector(".ueb-pin-name").getBoundingClientRect().width - ); - this.inputContentElements.forEach(inputElement => this.#updateWrapClass(inputElement)); - } - } - - setup() { - super.setup(); - this.#inputContentElements.forEach(element => { - element.addEventListener("focusout", this.#onFocusOutHandler); - if (/** @type {typeof IInputPinTemplate} */(this.constructor).canWrapInput) { - element.addEventListener("input", this.#onInputCheckWrapHandler); - } - }); - } - - cleanup() { - super.cleanup(); - this.#inputContentElements.forEach(element => { - element.removeEventListener("focusout", this.#onFocusOutHandler); - element.removeEventListener("input", this.#onInputCheckWrapHandler); - }); - } - - createInputObjects() { - return [ - ...super.createInputObjects(), - ...this.#inputContentElements.map(elem => new MouseIgnore(elem, this.blueprint)), - ] - } - - getInput() { - return this.getInputs().reduce((acc, cur) => acc + cur, "") - } - - getInputs() { - return this.#inputContentElements.map(element => - // Faster than innerText which causes reflow - Utility.clearHTMLWhitespace(element.innerHTML) - ) - } - - /** @param {String[]} values */ - setInputs(values = [], updateDefaultValue = true) { - this.#inputContentElements.forEach(/** @type {typeof IInputPinTemplate } */(this.constructor).singleLineInput - ? (elem, i) => elem.innerText = values[i] - : (elem, i) => elem.innerText = values[i].replaceAll("\n", "") - ); - if (updateDefaultValue) { - this.setDefaultValue(values.map(v => IInputPinTemplate.stringFromInputToUE(v)), values); - } - this.element.addNextUpdatedCallbacks(() => this.element.nodeElement.acknowledgeReflow()); - } - - setDefaultValue(values = [], rawValues = values) { - this.element.setDefaultValue( - // @ts-expect-error - values.join("") - ); - } - - renderInput() { - const singleLine = /** @type {typeof IInputPinTemplate} */(this.constructor).singleLineInput; - const selectOnFocus = /** @type {typeof IInputPinTemplate} */(this.constructor).selectOnFocus; - return y` -
- - -
- ` - } -} - -/** - * @template T - * @extends IInputPinTemplate - */ -class INumericPinTemplate extends IInputPinTemplate { - - static singleLineInput = true - - /** @param {String[]} values */ - setInputs(values = [], updateDefaultValue = false) { - if (!values || values.length == 0) { - values = [this.getInput()]; - } - super.setInputs(values, false); - if (updateDefaultValue) { - let parsedValues = []; - for (const value of values) { - let num = parseFloat(value); - if (isNaN(num)) { - num = 0; - updateDefaultValue = false; - } - parsedValues.push(num); - } - this.setDefaultValue(parsedValues, values); - } - } - - /** - * @param {Number[]} values - * @param {String[]} rawValues - */ - setDefaultValue(values = [], rawValues) { - this.element.setDefaultValue(/** @type {T} */(values[0])); - } -} - -/** @typedef {import("../../entity/IntegerEntity").default} IntegerEntity */ - -/** @extends INumericPinTemplate */ -class IntPinTemplate extends INumericPinTemplate { - - setDefaultValue(values = [], rawValues = values) { - const integer = this.element.getDefaultValue(true); - integer.value = values[0]; - this.inputContentElements[0].innerText = this.element.getDefaultValue()?.toString(); // needed - this.element.requestUpdate(); - } - - renderInput() { - return y` -
- - -
- ` - } -} - -/** @typedef {import("../../entity/IntegerEntity").default} IntegerEntity */ - -class Int64PinTemplate extends IntPinTemplate { - - /** @param {String[]} values */ - setInputs(values = [], updateDefaultValue = false) { - if (!values || values.length == 0) { - values = [this.getInput()]; - } - super.setInputs(values, false); - if (updateDefaultValue) { - if (!values[0].match(/[\-\+]?[0-9]+/)) { - return - } - const parsedValues = [BigInt(values[0])]; - this.setDefaultValue(parsedValues, values); - } - } -} - -/** - * @license - * Copyright 2017 Google LLC - * SPDX-License-Identifier: BSD-3-Clause - */ -const t={ATTRIBUTE:1,CHILD:2,PROPERTY:3,BOOLEAN_ATTRIBUTE:4,EVENT:5,ELEMENT:6},e=t=>(...e)=>({_$litDirective$:t,values:e});class i$1{constructor(t){}get _$AU(){return this._$AM._$AU}_$AT(t,e,i){this._$Ct=t,this._$AM=e,this._$Ci=i;}_$AS(t,e){return this.update(t,e)}update(t,e){return this.render(...e)}} - -/** - * @license - * Copyright 2018 Google LLC - * SPDX-License-Identifier: BSD-3-Clause - */const i=e(class extends i$1{constructor(t$1){var e;if(super(t$1),t$1.type!==t.ATTRIBUTE||"style"!==t$1.name||(null===(e=t$1.strings)||void 0===e?void 0:e.length)>2)throw Error("The `styleMap` directive must be used in the `style` attribute and must be the only part in the attribute.")}render(t){return Object.keys(t).reduce(((e,r)=>{const s=t[r];return null==s?e:e+`${r=r.replace(/(?:^(webkit|moz|ms|o)|)(?=[A-Z])/g,"-$&").toLowerCase()}:${s};`}),"")}update(e,[r]){const{style:s}=e.element;if(void 0===this.vt){this.vt=new Set;for(const t in r)this.vt.add(t);return this.render(r)}this.vt.forEach((t=>{null==r[t]&&(this.vt.delete(t),t.includes("-")?s.removeProperty(t):s[t]="");}));for(const t in r){const e=r[t];null!=e&&(this.vt.add(t),t.includes("-")?s.setProperty(t,e):s[t]=e);}return x}}); - +} + +/** @extends {IDraggableControlElement} */ +class ColorSliderElement extends IDraggableControlElement { + + constructor() { + super(); + super.initialize({}, new ColorSliderTemplate()); + } + + static newObject() { + return new ColorSliderElement() + } + + initialize() { + // Initialized in the constructor, this method does nothing + } +} + +/** @typedef {import ("../../element/InputElement").default} InputElement */ + +/** @extends {ITemplate} */ +class InputTemplate extends ITemplate { + + #focusHandler = () => { + this.blueprint.acknowledgeEditText(true); + if (this.element.selectOnFocus) { + getSelection().selectAllChildren(this.element); + } + } + + #focusoutHandler = () => { + this.blueprint.acknowledgeEditText(false); + getSelection().removeAllRanges(); // Deselect eventually selected text inside the input + } + + /** @param {InputEvent} e */ + #inputSingleLineHandler = e => + /** @type {HTMLElement} */(e.target).querySelectorAll("br").forEach(br => br.remove()) + + /** @param {KeyboardEvent} e */ + #onKeydownBlurOnEnterHandler = e => { + if (e.code == "Enter" && !e.shiftKey) { + /** @type {HTMLElement} */(e.target).blur(); + } + } + + /** @param {InputElement} element */ + initialize(element) { + super.initialize(element); + this.element.classList.add("ueb-pin-input-content"); + this.element.setAttribute("role", "textbox"); + this.element.contentEditable = "true"; + } + + setup() { + super.setup(); + this.element.addEventListener("focus", this.#focusHandler); + this.element.addEventListener("focusout", this.#focusoutHandler); + if (this.element.singleLine) { + this.element.addEventListener("input", this.#inputSingleLineHandler); + } + if (this.element.blurOnEnter) { + this.element.addEventListener("keydown", this.#onKeydownBlurOnEnterHandler); + } + } + + cleanup() { + super.cleanup(); + this.element.removeEventListener("focus", this.#focusHandler); + this.element.removeEventListener("focusout", this.#focusoutHandler); + this.element.removeEventListener("input", this.#inputSingleLineHandler); + this.element.removeEventListener("keydown", this.#onKeydownBlurOnEnterHandler); + } +} + +class InputElement extends IElement { + + static properties = { + ...super.properties, + singleLine: { + type: Boolean, + attribute: "data-single-line", + converter: Utility.booleanConverter, + reflect: true, + }, + selectOnFocus: { + type: Boolean, + attribute: "data-select-focus", + converter: Utility.booleanConverter, + reflect: true, + }, + blurOnEnter: { + type: Boolean, + attribute: "data-blur-enter", + converter: Utility.booleanConverter, + reflect: true, + }, + } + + constructor() { + super(); + this.singleLine = false; + this.selectOnFocus = true; + this.blurOnEnter = true; + super.initialize({}, new InputTemplate()); + } + + static newObject() { + return new InputElement() + } + + initialize() { + // Initialized in the constructor, this method does nothing + } +} + +/** + * @typedef {import("../../element/IDraggableElement").default} IDraggableElement + */ + +/** +* @template {IDraggableElement} T +* @extends {IMouseClickDrag} +*/ +class MouseIgnore extends IMouseClickDrag { + + constructor(target, blueprint, options = {}) { + options.consumeEvent = true; + super(target, blueprint, options); + } +} + +/** @typedef {import("lit").PropertyValues} PropertyValues */ + +/** @extends PinTemplate */ +class BoolPinTemplate extends PinTemplate { + + /** @type {HTMLInputElement?} */ + #input + + #onChangeHandler = _ => this.element.setDefaultValue(this.#input.checked) + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + this.#input = this.element.querySelector(".ueb-pin-input"); + } + + setup() { + super.setup(); + this.#input?.addEventListener("change", this.#onChangeHandler); + } + + cleanup() { + super.cleanup(); + this.#input?.removeEventListener("change", this.#onChangeHandler); + } + + createInputObjects() { + return [ + ...super.createInputObjects(), + new MouseIgnore(this.#input, this.blueprint), + ] + } + + renderInput() { + return y` + + ` + } +} + +/** @typedef {import("../../element/PinElement").default} PinElement */ + +class ExecPinTemplate extends PinTemplate { + + renderIcon() { + return SVGIcon.execPin + } + + renderName() { + let pinName = this.element.entity.PinName; + if (this.element.entity.PinFriendlyName) { + pinName = this.element.entity.PinFriendlyName.toString(); + } else if (pinName === "execute" || pinName === "then") { + return y`` + } + return y`${Utility.formatStringName(pinName)}` + } +} + +/** @typedef {import("lit").PropertyValues} PropertyValues */ + +/** + * @template T + * @extends PinTemplate + */ +class IInputPinTemplate extends PinTemplate { + + static singleLineInput = false + static selectOnFocus = true + + /** @type {HTMLElement[]} */ + #inputContentElements + get inputContentElements() { + return this.#inputContentElements + } + + /** @param {String} value */ + static stringFromInputToUE(value) { + return value + .replace(/(?=\n\s*)\n$/, "") // Remove trailing double newline + .replaceAll("\n", "\\r\n") // Replace newline with \r\n (default newline in UE) + } + + /** @param {String} value */ + static stringFromUEToInput(value) { + return value + .replaceAll(/(?:\r|(?<=(?:^|[^\\])(?:\\\\)*)\\r)(?=\n)/g, "") // Remove \r leftover from \r\n + .replace(/(?<=\n\s*)$/, "\n") // Put back trailing double newline + } + + #onFocusOutHandler = () => this.setInputs(this.getInputs(), true) + /** @param {InputEvent} event */ + #onInputCheckWrapHandler = event => this.#updateWrapClass(/** @type {HTMLElement} */(event.target)) + + /** @param {HTMLElement} inputElement*/ + #updateWrapClass(inputElement) { + const width = this.blueprint.scaleCorrect(inputElement.getBoundingClientRect().width) + this.nameWidth; + const inputWrapped = this.element.classList.contains("ueb-pin-input-wrap"); + if (!inputWrapped && width > Configuration.pinInputWrapWidth) { + this.element.classList.add("ueb-pin-input-wrap"); + } else if (inputWrapped && width <= Configuration.pinInputWrapWidth) { + this.element.classList.remove("ueb-pin-input-wrap"); + } + } + + /** @param {PropertyValues} changedProperties */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + this.#inputContentElements = /** @type {HTMLElement[]} */([...this.element.querySelectorAll("ueb-input")]); + if (/** @type {typeof IInputPinTemplate} */(this.constructor).canWrapInput) { + this.nameWidth = this.blueprint.scaleCorrect( + this.element.querySelector(".ueb-pin-name").getBoundingClientRect().width + ); + this.inputContentElements.forEach(inputElement => this.#updateWrapClass(inputElement)); + } + } + + setup() { + super.setup(); + this.#inputContentElements.forEach(element => { + element.addEventListener("focusout", this.#onFocusOutHandler); + if (/** @type {typeof IInputPinTemplate} */(this.constructor).canWrapInput) { + element.addEventListener("input", this.#onInputCheckWrapHandler); + } + }); + } + + cleanup() { + super.cleanup(); + this.#inputContentElements.forEach(element => { + element.removeEventListener("focusout", this.#onFocusOutHandler); + element.removeEventListener("input", this.#onInputCheckWrapHandler); + }); + } + + createInputObjects() { + return [ + ...super.createInputObjects(), + ...this.#inputContentElements.map(elem => new MouseIgnore(elem, this.blueprint)), + ] + } + + getInput() { + return this.getInputs().reduce((acc, cur) => acc + cur, "") + } + + getInputs() { + return this.#inputContentElements.map(element => + // Faster than innerText which causes reflow + Utility.clearHTMLWhitespace(element.innerHTML) + ) + } + + /** @param {String[]} values */ + setInputs(values = [], updateDefaultValue = true) { + this.#inputContentElements.forEach(/** @type {typeof IInputPinTemplate } */(this.constructor).singleLineInput + ? (elem, i) => elem.innerText = values[i] + : (elem, i) => elem.innerText = values[i].replaceAll("\n", "") + ); + if (updateDefaultValue) { + this.setDefaultValue(values.map(v => IInputPinTemplate.stringFromInputToUE(v)), values); + } + this.element.addNextUpdatedCallbacks(() => this.element.nodeElement.acknowledgeReflow()); + } + + setDefaultValue(values = [], rawValues = values) { + this.element.setDefaultValue( + // @ts-expect-error + values.join("") + ); + } + + renderInput() { + const singleLine = /** @type {typeof IInputPinTemplate} */(this.constructor).singleLineInput; + const selectOnFocus = /** @type {typeof IInputPinTemplate} */(this.constructor).selectOnFocus; + return y` +
+ + +
+ ` + } +} + +/** + * @template T + * @extends IInputPinTemplate + */ +class INumericPinTemplate extends IInputPinTemplate { + + static singleLineInput = true + + /** @param {String[]} values */ + setInputs(values = [], updateDefaultValue = false) { + if (!values || values.length == 0) { + values = [this.getInput()]; + } + super.setInputs(values, false); + if (updateDefaultValue) { + let parsedValues = []; + for (const value of values) { + let num = parseFloat(value); + if (isNaN(num)) { + num = 0; + updateDefaultValue = false; + } + parsedValues.push(num); + } + this.setDefaultValue(parsedValues, values); + } + } + + /** + * @param {Number[]} values + * @param {String[]} rawValues + */ + setDefaultValue(values = [], rawValues) { + this.element.setDefaultValue(/** @type {T} */(values[0])); + } +} + +/** @typedef {import("../../entity/IntegerEntity").default} IntegerEntity */ + +/** @extends INumericPinTemplate */ +class IntPinTemplate extends INumericPinTemplate { + + setDefaultValue(values = [], rawValues = values) { + const integer = this.element.getDefaultValue(true); + integer.value = values[0]; + this.inputContentElements[0].innerText = this.element.getDefaultValue()?.toString(); // needed + this.element.requestUpdate(); + } + + renderInput() { + return y` +
+ + +
+ ` + } +} + +/** @typedef {import("../../entity/IntegerEntity").default} IntegerEntity */ + +class Int64PinTemplate extends IntPinTemplate { + + /** @param {String[]} values */ + setInputs(values = [], updateDefaultValue = false) { + if (!values || values.length == 0) { + values = [this.getInput()]; + } + super.setInputs(values, false); + if (updateDefaultValue) { + if (!values[0].match(/[\-\+]?[0-9]+/)) { + return + } + const parsedValues = [BigInt(values[0])]; + this.setDefaultValue(parsedValues, values); + } + } +} + +/** + * @license + * Copyright 2017 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + */ +const t={ATTRIBUTE:1,CHILD:2,PROPERTY:3,BOOLEAN_ATTRIBUTE:4,EVENT:5,ELEMENT:6},e=t=>(...e)=>({_$litDirective$:t,values:e});class i$1{constructor(t){}get _$AU(){return this._$AM._$AU}_$AT(t,e,i){this._$Ct=t,this._$AM=e,this._$Ci=i;}_$AS(t,e){return this.update(t,e)}update(t,e){return this.render(...e)}} + +/** + * @license + * Copyright 2018 Google LLC + * SPDX-License-Identifier: BSD-3-Clause + */const i=e(class extends i$1{constructor(t$1){var e;if(super(t$1),t$1.type!==t.ATTRIBUTE||"style"!==t$1.name||(null===(e=t$1.strings)||void 0===e?void 0:e.length)>2)throw Error("The `styleMap` directive must be used in the `style` attribute and must be the only part in the attribute.")}render(t){return Object.keys(t).reduce(((e,r)=>{const s=t[r];return null==s?e:e+`${r=r.replace(/(?:^(webkit|moz|ms|o)|)(?=[A-Z])/g,"-$&").toLowerCase()}:${s};`}),"")}update(e,[r]){const{style:s}=e.element;if(void 0===this.vt){this.vt=new Set;for(const t in r)this.vt.add(t);return this.render(r)}this.vt.forEach((t=>{null==r[t]&&(this.vt.delete(t),t.includes("-")?s.removeProperty(t):s[t]="");}));for(const t in r){const e=r[t];null!=e&&(this.vt.add(t),t.includes("-")?s.setProperty(t,e):s[t]=e);}return x}}); + /** @typedef {import("../../element/WindowElement").default} WindowElement */ /** @extends {IDraggablePositionedTemplate} */ @@ -8483,8 +8523,8 @@ class WindowTemplate extends IDraggablePositionedTemplate { this.element.dispatchEvent(new CustomEvent(Configuration.windowCancelEventName)); this.element.remove(); } -} - +} + /** * @typedef {import("../../element/WindowElement").default} WindowElement * @typedef {import("lit").PropertyValues} PropertyValues @@ -8838,501 +8878,501 @@ class ColorPickerWindowTemplate extends WindowTemplate { renderWindowName() { return y`${Configuration.colorWindowName}` } -} - -/** - * @typedef {import("../../element/WindowElement").default} WindowElement - * @typedef {import("../../element/WindowElement").WindowElementConstructor} WindowElementConstructor - * @typedef {import("../../entity/LinearColorEntity").default} LinearColorEntity - */ - -/** @extends PinTemplate */ -class LinearColorPinTemplate extends PinTemplate { - - /** @type {WindowElement} */ - #window - - /** @param {MouseEvent} e */ - #launchColorPickerWindow = e => { - e.preventDefault(); - this.blueprint.setFocused(true); - /** @type {WindowElement} */ - this.#window = /** @type {WindowElementConstructor} */(ElementFactory.getConstructor("ueb-window")) - .newObject({ - type: new ColorPickerWindowTemplate(), - windowOptions: { - // The created window will use the following functions to get and set the color - getPinColor: () => this.element.defaultValue, - /** @param {LinearColorEntity} color */ - setPinColor: color => this.element.setDefaultValue(color), - }, - }); - this.blueprint.append(this.#window); - const windowApplyHandler = () => { - this.element.setDefaultValue( - /** @type {ColorPickerWindowTemplate} */(this.#window.template).color - ); - }; - const windowCloseHandler = () => { - this.#window.removeEventListener(Configuration.windowApplyEventName, windowApplyHandler); - this.#window.removeEventListener(Configuration.windowCloseEventName, windowCloseHandler); - this.#window = null; - }; - this.#window.addEventListener(Configuration.windowApplyEventName, windowApplyHandler); - this.#window.addEventListener(Configuration.windowCloseEventName, windowCloseHandler); - } - - renderInput() { - return y` - - - ` - } -} - -/** @typedef {import("../../element/PinElement").default} PinElement */ - -class NamePinTemplate extends IInputPinTemplate { - - static singleLineInput = true -} - -/** - * @template {Number} T - * @extends INumericPinTemplate - */ -class RealPinTemplate extends INumericPinTemplate { - - setDefaultValue(values = [], rawValues = values) { - this.element.setDefaultValue(values[0]); - } - - renderInput() { - return y` -
- - -
- ` - } -} - -class ReferencePinTemplate extends PinTemplate { - - renderIcon() { - return SVGIcon.referencePin - } -} - -/** @typedef {import("../../entity/RotatorEntity").default} Rotator */ - -/** @extends INumericPinTemplate */ -class RotatorPinTemplate extends INumericPinTemplate { - - #getR() { - return Utility.minDecimals(this.element.getDefaultValue()?.R ?? 0) - } - - #getP() { - return Utility.minDecimals(this.element.getDefaultValue()?.P ?? 0) - } - - #getY() { - return Utility.minDecimals(this.element.getDefaultValue()?.Y ?? 0) - } - - setDefaultValue(values = [], rawValues = values) { - const rotator = this.element.getDefaultValue(true); - if (!(rotator instanceof RotatorEntity)) { - throw new TypeError("Expected DefaultValue to be a RotatorEntity") - } - rotator.R = values[0]; // Roll - rotator.P = values[1]; // Pitch - rotator.Y = values[2]; // Yaw - this.element.requestUpdate("DefaultValue", rotator); - } - - renderInput() { - return y` -
- X -
- -
- Y -
- -
- Z -
- -
-
- ` - } -} - -/** @extends IInputPinTemplate */ -class StringPinTemplate extends IInputPinTemplate { -} - -/** - * @extends INumericPinTemplate - */ -class VectorInputPinTemplate extends INumericPinTemplate { - - #getX() { - return Utility.minDecimals(this.element.getDefaultValue()?.X ?? 0) - } - - #getY() { - return Utility.minDecimals(this.element.getDefaultValue()?.Y ?? 0) - } - - /** - * @param {Number[]} values - * @param {String[]} rawValues - */ - setDefaultValue(values, rawValues) { - const vector = this.element.getDefaultValue(true); - if (!(vector instanceof Vector2DEntity)) { - throw new TypeError("Expected DefaultValue to be a Vector2DEntity") - } - vector.X = values[0]; - vector.Y = values[1]; - this.element.requestUpdate("DefaultValue", vector); - } - - renderInput() { - return y` -
- X -
- -
- Y -
- -
-
- ` - } -} - -/** - * @extends INumericPinTemplate - */ -class VectorPinTemplate extends INumericPinTemplate { - - #getX() { - return Utility.minDecimals(this.element.getDefaultValue()?.X ?? 0) - } - - #getY() { - return Utility.minDecimals(this.element.getDefaultValue()?.Y ?? 0) - } - - #getZ() { - return Utility.minDecimals(this.element.getDefaultValue()?.Z ?? 0) - } - - /** - * @param {Number[]} values - * @param {String[]} rawValues - */ - setDefaultValue(values, rawValues) { - const vector = this.element.getDefaultValue(true); - if (!(vector instanceof VectorEntity)) { - throw new TypeError("Expected DefaultValue to be a VectorEntity") - } - vector.X = values[0]; - vector.Y = values[1]; - vector.Z = values[2]; - this.element.requestUpdate("DefaultValue", vector); - } - - renderInput() { - return y` -
- X -
- -
- Y -
- -
- Z -
- -
-
- ` - } -} - -/** - * @typedef {import("../entity/IEntity").AnyValue} AnyValue - * @typedef {import("./LinkElement").LinkElementConstructor} LinkElementConstructor - * @typedef {import("./NodeElement").default} NodeElement - * @typedef {import("lit").CSSResult} CSSResult - * @typedef {typeof PinElement} PinElementConstructor - */ - -/** - * @template {AnyValue} T - * @extends {IElement, PinTemplate>} - */ -class PinElement extends IElement { - - static #inputPinTemplates = { - "/Script/CoreUObject.LinearColor": LinearColorPinTemplate, - "/Script/CoreUObject.Rotator": RotatorPinTemplate, - "/Script/CoreUObject.Vector": VectorPinTemplate, - "/Script/CoreUObject.Vector2D": VectorInputPinTemplate, - "bool": BoolPinTemplate, - "byte": IntPinTemplate, - "int": IntPinTemplate, - "int64": Int64PinTemplate, - "MUTABLE_REFERENCE": ReferencePinTemplate, - "name": NamePinTemplate, - "real": RealPinTemplate, - "string": StringPinTemplate, - } - - static properties = { - pinId: { - type: GuidEntity, - converter: { - fromAttribute: (value, type) => value - ? ISerializer.grammar.Guid.parse(value).value - : null, - toAttribute: (value, type) => value?.toString(), - }, - attribute: "data-id", - reflect: true, - }, - pinType: { - type: String, - attribute: "data-type", - reflect: true, - }, - advancedView: { - type: String, - attribute: "data-advanced-view", - reflect: true, - }, - color: { - type: LinearColorEntity, - converter: { - fromAttribute: (value, type) => value - ? ISerializer.grammar.LinearColorFromAnyColor.parse(value).value - : null, - toAttribute: (value, type) => value ? Utility.printLinearColor(value) : null, - }, - attribute: "data-color", - reflect: true, - }, - defaultValue: { - type: String, - attribute: false, - }, - isLinked: { - type: Boolean, - converter: Utility.booleanConverter, - attribute: "data-linked", - reflect: true, - }, - pinDirection: { - type: String, - attribute: "data-direction", - reflect: true, - }, - } - - /** @type {NodeElement} */ - nodeElement - - /** - * @param {PinEntity} pinEntity - * @return {new () => PinTemplate} - */ - static getTypeTemplate(pinEntity) { - if (pinEntity.PinType.bIsReference && !pinEntity.PinType.bIsConst) { - return PinElement.#inputPinTemplates["MUTABLE_REFERENCE"] - } - if (pinEntity.getType() === "exec") { - return ExecPinTemplate - } - let result; - if (pinEntity.isInput()) { - result = PinElement.#inputPinTemplates[pinEntity.getType()]; - } - return result ?? PinTemplate - } - - static newObject( - entity = new PinEntity(), - template = new (PinElement.getTypeTemplate(entity))(), - nodeElement = undefined - ) { - const result = new PinElement(); - result.initialize(entity, template, nodeElement); - return result - } - - initialize( - entity = /** @type {PinEntity} */(new PinEntity()), - template = new (PinElement.getTypeTemplate(entity))(), - nodeElement = undefined - ) { - super.initialize(entity, template); - this.pinId = this.entity.PinId; - this.pinType = this.entity.getType(); - this.advancedView = this.entity.bAdvancedView; - this.defaultValue = this.entity.getDefaultValue(); - this.color = PinElement.properties.color.converter.fromAttribute(this.getColor().toString()); - this.isLinked = false; - this.pinDirection = entity.isInput() ? "input" : entity.isOutput() ? "output" : "hidden"; - this.nodeElement = /** @type {NodeElement} */(nodeElement); - } - - setup() { - super.setup(); - this.nodeElement = this.closest("ueb-node"); - } - - createPinReference() { - return new PinReferenceEntity({ - objectName: this.nodeElement.getNodeName(), - pinGuid: this.getPinId(), - }) - } - - /** @return {GuidEntity} */ - getPinId() { - return this.entity.PinId - } - - /** @returns {String} */ - getPinName() { - return this.entity.PinName - } - - getPinDisplayName() { - return this.entity.getDisplayName() - } - - /** @return {CSSResult} */ - getColor() { - return this.entity.pinColor() - } - - isInput() { - return this.entity.isInput() - } - - isOutput() { - return this.entity.isOutput() - } - - getLinkLocation() { - return this.template.getLinkLocation() - } - - getNodeElement() { - return this.nodeElement - } - - getLinks() { - return this.entity.LinkedTo ?? [] - } - - getDefaultValue(maybeCreate = false) { - return this.defaultValue = this.entity.getDefaultValue(maybeCreate) - } - - /** @param {T} value */ - setDefaultValue(value) { - this.entity.DefaultValue = value; - this.defaultValue = value; - } - - /** @param {IElement[]} nodesWhitelist */ - sanitizeLinks(nodesWhitelist = []) { - this.entity.LinkedTo = this.entity.LinkedTo?.filter(pinReference => { - let pin = this.blueprint.getPin(pinReference); - if (pin) { - if (nodesWhitelist.length && !nodesWhitelist.includes(pin.nodeElement)) { - return false - } - let link = this.blueprint.getLink(this, pin); - if (!link) { - link = /** @type {LinkElementConstructor} */(ElementFactory.getConstructor("ueb-link")) - .newObject(this, pin); - this.blueprint.addGraphElement(link); - } - } - return pin - }); - this.isLinked = this.entity.isLinked(); - } - - /** @param {PinElement} targetPinElement */ - linkTo(targetPinElement) { - const pinReference = this.createPinReference(); - if ( - this.isLinked - && this.isOutput() - && (this.pinType === "exec" || targetPinElement.pinType === "exec") - && !this.getLinks().some(ref => pinReference.equals(ref))) { - this.unlinkFromAll(); - } - if (this.entity.linkTo(targetPinElement.getNodeElement().getNodeName(), targetPinElement.entity)) { - this.isLinked = this.entity.isLinked(); - this.nodeElement?.template.linksChanged(); - } - } - - /** @param {PinElement} targetPinElement */ - unlinkFrom(targetPinElement, removeLink = true) { - if (this.entity.unlinkFrom(targetPinElement.getNodeElement().getNodeName(), targetPinElement.entity)) { - this.isLinked = this.entity.isLinked(); - this.nodeElement?.template.linksChanged(); - if (removeLink) { - this.blueprint.getLink(this, targetPinElement)?.remove(); // Might be called after the link is removed - } - } - } - - unlinkFromAll() { - const isLinked = this.getLinks().length; - this.getLinks().map(ref => this.blueprint.getPin(ref)).forEach(pin => this.unlinkFrom(pin)); - if (isLinked) { - this.nodeElement?.template.linksChanged(); - } - } - - /** - * @param {PinElement} originalPinElement - * @param {PinReferenceEntity} newReference - */ - redirectLink(originalPinElement, newReference) { - const index = this.getLinks().findIndex(pinReference => - pinReference.objectName.toString() == originalPinElement.getNodeElement().getNodeName() - && pinReference.pinGuid.valueOf() == originalPinElement.entity.PinId.valueOf() - ); - if (index >= 0) { - this.entity.LinkedTo[index] = newReference; - return true - } - return false - } -} - +} + +/** + * @typedef {import("../../element/WindowElement").default} WindowElement + * @typedef {import("../../element/WindowElement").WindowElementConstructor} WindowElementConstructor + * @typedef {import("../../entity/LinearColorEntity").default} LinearColorEntity + */ + +/** @extends PinTemplate */ +class LinearColorPinTemplate extends PinTemplate { + + /** @type {WindowElement} */ + #window + + /** @param {MouseEvent} e */ + #launchColorPickerWindow = e => { + e.preventDefault(); + this.blueprint.setFocused(true); + /** @type {WindowElement} */ + this.#window = /** @type {WindowElementConstructor} */(ElementFactory.getConstructor("ueb-window")) + .newObject({ + type: new ColorPickerWindowTemplate(), + windowOptions: { + // The created window will use the following functions to get and set the color + getPinColor: () => this.element.defaultValue, + /** @param {LinearColorEntity} color */ + setPinColor: color => this.element.setDefaultValue(color), + }, + }); + this.blueprint.append(this.#window); + const windowApplyHandler = () => { + this.element.setDefaultValue( + /** @type {ColorPickerWindowTemplate} */(this.#window.template).color + ); + }; + const windowCloseHandler = () => { + this.#window.removeEventListener(Configuration.windowApplyEventName, windowApplyHandler); + this.#window.removeEventListener(Configuration.windowCloseEventName, windowCloseHandler); + this.#window = null; + }; + this.#window.addEventListener(Configuration.windowApplyEventName, windowApplyHandler); + this.#window.addEventListener(Configuration.windowCloseEventName, windowCloseHandler); + } + + renderInput() { + return y` + + + ` + } +} + +/** @typedef {import("../../element/PinElement").default} PinElement */ + +class NamePinTemplate extends IInputPinTemplate { + + static singleLineInput = true +} + +/** + * @template {Number} T + * @extends INumericPinTemplate + */ +class RealPinTemplate extends INumericPinTemplate { + + setDefaultValue(values = [], rawValues = values) { + this.element.setDefaultValue(values[0]); + } + + renderInput() { + return y` +
+ + +
+ ` + } +} + +class ReferencePinTemplate extends PinTemplate { + + renderIcon() { + return SVGIcon.referencePin + } +} + +/** @typedef {import("../../entity/RotatorEntity").default} Rotator */ + +/** @extends INumericPinTemplate */ +class RotatorPinTemplate extends INumericPinTemplate { + + #getR() { + return Utility.minDecimals(this.element.getDefaultValue()?.R ?? 0) + } + + #getP() { + return Utility.minDecimals(this.element.getDefaultValue()?.P ?? 0) + } + + #getY() { + return Utility.minDecimals(this.element.getDefaultValue()?.Y ?? 0) + } + + setDefaultValue(values = [], rawValues = values) { + const rotator = this.element.getDefaultValue(true); + if (!(rotator instanceof RotatorEntity)) { + throw new TypeError("Expected DefaultValue to be a RotatorEntity") + } + rotator.R = values[0]; // Roll + rotator.P = values[1]; // Pitch + rotator.Y = values[2]; // Yaw + this.element.requestUpdate("DefaultValue", rotator); + } + + renderInput() { + return y` +
+ X +
+ +
+ Y +
+ +
+ Z +
+ +
+
+ ` + } +} + +/** @extends IInputPinTemplate */ +class StringPinTemplate extends IInputPinTemplate { +} + +/** + * @extends INumericPinTemplate + */ +class VectorInputPinTemplate extends INumericPinTemplate { + + #getX() { + return Utility.minDecimals(this.element.getDefaultValue()?.X ?? 0) + } + + #getY() { + return Utility.minDecimals(this.element.getDefaultValue()?.Y ?? 0) + } + + /** + * @param {Number[]} values + * @param {String[]} rawValues + */ + setDefaultValue(values, rawValues) { + const vector = this.element.getDefaultValue(true); + if (!(vector instanceof Vector2DEntity)) { + throw new TypeError("Expected DefaultValue to be a Vector2DEntity") + } + vector.X = values[0]; + vector.Y = values[1]; + this.element.requestUpdate("DefaultValue", vector); + } + + renderInput() { + return y` +
+ X +
+ +
+ Y +
+ +
+
+ ` + } +} + +/** + * @extends INumericPinTemplate + */ +class VectorPinTemplate extends INumericPinTemplate { + + #getX() { + return Utility.minDecimals(this.element.getDefaultValue()?.X ?? 0) + } + + #getY() { + return Utility.minDecimals(this.element.getDefaultValue()?.Y ?? 0) + } + + #getZ() { + return Utility.minDecimals(this.element.getDefaultValue()?.Z ?? 0) + } + + /** + * @param {Number[]} values + * @param {String[]} rawValues + */ + setDefaultValue(values, rawValues) { + const vector = this.element.getDefaultValue(true); + if (!(vector instanceof VectorEntity)) { + throw new TypeError("Expected DefaultValue to be a VectorEntity") + } + vector.X = values[0]; + vector.Y = values[1]; + vector.Z = values[2]; + this.element.requestUpdate("DefaultValue", vector); + } + + renderInput() { + return y` +
+ X +
+ +
+ Y +
+ +
+ Z +
+ +
+
+ ` + } +} + +/** + * @typedef {import("../entity/IEntity").AnyValue} AnyValue + * @typedef {import("./LinkElement").LinkElementConstructor} LinkElementConstructor + * @typedef {import("./NodeElement").default} NodeElement + * @typedef {import("lit").CSSResult} CSSResult + * @typedef {typeof PinElement} PinElementConstructor + */ + +/** + * @template {AnyValue} T + * @extends {IElement, PinTemplate>} + */ +class PinElement extends IElement { + + static #inputPinTemplates = { + "/Script/CoreUObject.LinearColor": LinearColorPinTemplate, + "/Script/CoreUObject.Rotator": RotatorPinTemplate, + "/Script/CoreUObject.Vector": VectorPinTemplate, + "/Script/CoreUObject.Vector2D": VectorInputPinTemplate, + "bool": BoolPinTemplate, + "byte": IntPinTemplate, + "int": IntPinTemplate, + "int64": Int64PinTemplate, + "MUTABLE_REFERENCE": ReferencePinTemplate, + "name": NamePinTemplate, + "real": RealPinTemplate, + "string": StringPinTemplate, + } + + static properties = { + pinId: { + type: GuidEntity, + converter: { + fromAttribute: (value, type) => value + ? ISerializer.grammar.Guid.parse(value).value + : null, + toAttribute: (value, type) => value?.toString(), + }, + attribute: "data-id", + reflect: true, + }, + pinType: { + type: String, + attribute: "data-type", + reflect: true, + }, + advancedView: { + type: String, + attribute: "data-advanced-view", + reflect: true, + }, + color: { + type: LinearColorEntity, + converter: { + fromAttribute: (value, type) => value + ? ISerializer.grammar.LinearColorFromAnyColor.parse(value).value + : null, + toAttribute: (value, type) => value ? Utility.printLinearColor(value) : null, + }, + attribute: "data-color", + reflect: true, + }, + defaultValue: { + type: String, + attribute: false, + }, + isLinked: { + type: Boolean, + converter: Utility.booleanConverter, + attribute: "data-linked", + reflect: true, + }, + pinDirection: { + type: String, + attribute: "data-direction", + reflect: true, + }, + } + + /** @type {NodeElement} */ + nodeElement + + /** + * @param {PinEntity} pinEntity + * @return {new () => PinTemplate} + */ + static getTypeTemplate(pinEntity) { + if (pinEntity.PinType.bIsReference && !pinEntity.PinType.bIsConst) { + return PinElement.#inputPinTemplates["MUTABLE_REFERENCE"] + } + if (pinEntity.getType() === "exec") { + return ExecPinTemplate + } + let result; + if (pinEntity.isInput()) { + result = PinElement.#inputPinTemplates[pinEntity.getType()]; + } + return result ?? PinTemplate + } + + static newObject( + entity = new PinEntity(), + template = new (PinElement.getTypeTemplate(entity))(), + nodeElement = undefined + ) { + const result = new PinElement(); + result.initialize(entity, template, nodeElement); + return result + } + + initialize( + entity = /** @type {PinEntity} */(new PinEntity()), + template = new (PinElement.getTypeTemplate(entity))(), + nodeElement = undefined + ) { + super.initialize(entity, template); + this.pinId = this.entity.PinId; + this.pinType = this.entity.getType(); + this.advancedView = this.entity.bAdvancedView; + this.defaultValue = this.entity.getDefaultValue(); + this.color = PinElement.properties.color.converter.fromAttribute(this.getColor().toString()); + this.isLinked = false; + this.pinDirection = entity.isInput() ? "input" : entity.isOutput() ? "output" : "hidden"; + this.nodeElement = /** @type {NodeElement} */(nodeElement); + } + + setup() { + super.setup(); + this.nodeElement = this.closest("ueb-node"); + } + + createPinReference() { + return new PinReferenceEntity({ + objectName: this.nodeElement.getNodeName(), + pinGuid: this.getPinId(), + }) + } + + /** @return {GuidEntity} */ + getPinId() { + return this.entity.PinId + } + + /** @returns {String} */ + getPinName() { + return this.entity.PinName + } + + getPinDisplayName() { + return this.entity.getDisplayName() + } + + /** @return {CSSResult} */ + getColor() { + return this.entity.pinColor() + } + + isInput() { + return this.entity.isInput() + } + + isOutput() { + return this.entity.isOutput() + } + + getLinkLocation() { + return this.template.getLinkLocation() + } + + getNodeElement() { + return this.nodeElement + } + + getLinks() { + return this.entity.LinkedTo ?? [] + } + + getDefaultValue(maybeCreate = false) { + return this.defaultValue = this.entity.getDefaultValue(maybeCreate) + } + + /** @param {T} value */ + setDefaultValue(value) { + this.entity.DefaultValue = value; + this.defaultValue = value; + } + + /** @param {IElement[]} nodesWhitelist */ + sanitizeLinks(nodesWhitelist = []) { + this.entity.LinkedTo = this.entity.LinkedTo?.filter(pinReference => { + let pin = this.blueprint.getPin(pinReference); + if (pin) { + if (nodesWhitelist.length && !nodesWhitelist.includes(pin.nodeElement)) { + return false + } + let link = this.blueprint.getLink(this, pin); + if (!link) { + link = /** @type {LinkElementConstructor} */(ElementFactory.getConstructor("ueb-link")) + .newObject(this, pin); + this.blueprint.addGraphElement(link); + } + } + return pin + }); + this.isLinked = this.entity.isLinked(); + } + + /** @param {PinElement} targetPinElement */ + linkTo(targetPinElement) { + const pinReference = this.createPinReference(); + if ( + this.isLinked + && this.isOutput() + && (this.pinType === "exec" || targetPinElement.pinType === "exec") + && !this.getLinks().some(ref => pinReference.equals(ref))) { + this.unlinkFromAll(); + } + if (this.entity.linkTo(targetPinElement.getNodeElement().getNodeName(), targetPinElement.entity)) { + this.isLinked = this.entity.isLinked(); + this.nodeElement?.template.linksChanged(); + } + } + + /** @param {PinElement} targetPinElement */ + unlinkFrom(targetPinElement, removeLink = true) { + if (this.entity.unlinkFrom(targetPinElement.getNodeElement().getNodeName(), targetPinElement.entity)) { + this.isLinked = this.entity.isLinked(); + this.nodeElement?.template.linksChanged(); + if (removeLink) { + this.blueprint.getLink(this, targetPinElement)?.remove(); // Might be called after the link is removed + } + } + } + + unlinkFromAll() { + const isLinked = this.getLinks().length; + this.getLinks().map(ref => this.blueprint.getPin(ref)).forEach(pin => this.unlinkFrom(pin)); + if (isLinked) { + this.nodeElement?.template.linksChanged(); + } + } + + /** + * @param {PinElement} originalPinElement + * @param {PinReferenceEntity} newReference + */ + redirectLink(originalPinElement, newReference) { + const index = this.getLinks().findIndex(pinReference => + pinReference.objectName.toString() == originalPinElement.getNodeElement().getNodeName() + && pinReference.pinGuid.valueOf() == originalPinElement.entity.PinId.valueOf() + ); + if (index >= 0) { + this.entity.LinkedTo[index] = newReference; + return true + } + return false + } +} + class OrderedIndexArray { /** @@ -9450,8 +9490,8 @@ class OrderedIndexArray { shiftRight(leftLimit, steps = 1) { this.array.set(this.array.subarray(leftLimit, -steps), leftLimit + steps); } -} - +} + /** * @typedef {import("../element/NodeElement").default} NodeElement * @typedef {typeof import("../Blueprint").default.nodeBoundariesSupplier} BoundariesFunction @@ -9610,141 +9650,141 @@ class FastSelectionModel { } this.finalPosition = finalPosition; } -} - +} + /** @typedef {import("../element/SelectorElement").default} SelectorElement */ /** @extends IFromToPositionedTemplate */ class SelectorTemplate extends IFromToPositionedTemplate { -} - -/** @typedef {import("../Blueprint").BlueprintConstructor} BlueprintConstructor */ - -/** @extends {IFromToPositionedElement} */ -class SelectorElement extends IFromToPositionedElement { - - /** @type {FastSelectionModel} */ - selectionModel = null - - constructor() { - super(); - super.initialize({}, new SelectorTemplate()); - } - - static newObject() { - return new SelectorElement() - } - - initialize() { - // Initialized in the constructor, this method does nothing - } - - /** @param {Number[]} initialPosition */ - beginSelect(initialPosition) { - const blueprintConstructor = /** @type {BlueprintConstructor} */(this.blueprint.constructor); - this.blueprint.selecting = true; - this.setBothLocations(initialPosition); - this.selectionModel = new FastSelectionModel( - initialPosition, - this.blueprint.getNodes(), - blueprintConstructor.nodeBoundariesSupplier, - blueprintConstructor.nodeSelectToggleFunction - ); - } - - /** @param {Number[]} finalPosition */ - selectTo(finalPosition) { - this.selectionModel.selectTo(finalPosition); - this.toX = finalPosition[0]; - this.toY = finalPosition[1]; - } - - endSelect() { - this.blueprint.selecting = false; - this.selectionModel = null; - this.fromX = 0; - this.fromY = 0; - this.toX = 0; - this.toY = 0; - } -} - -/** @typedef {typeof WindowElement} WindowElementConstructor */ - -/** - * @template {WindowTemplate} T - * @extends {IDraggableElement} - */ -class WindowElement extends IDraggableElement { - - static #typeTemplateMap = { - "window": WindowTemplate, - "color-picker": ColorPickerWindowTemplate, - } - - static properties = { - ...IDraggableElement.properties, - type: { - type: WindowTemplate, - attribute: "data-type", - reflect: true, - converter: { - fromAttribute: (value, type) => WindowElement.#typeTemplateMap[value], - toAttribute: (value, type) => - Object.entries(WindowElement.#typeTemplateMap).find(([k, v]) => value.constructor === v)?.[0], - }, - }, - } - - static newObject(entity = {}, template = entity.type ?? new WindowTemplate()) { - const result = new WindowElement(); - result.initialize(entity, template); - return result - } - - initialize(entity = {}, template = entity.type ?? new WindowTemplate()) { - entity.windowOptions ??= {}; - this.type = entity.type; - this.windowOptions = entity.windowOptions; - super.initialize(entity, template); - } - - computeSizes() { - const bounding = this.getBoundingClientRect(); - this.sizeX = bounding.width; - this.sizeY = bounding.height; - } - - cleanup() { - super.cleanup(); - this.acknowledgeClose(); - } - - acknowledgeClose() { - let deleteEvent = new CustomEvent(Configuration.windowCloseEventName); - this.dispatchEvent(deleteEvent); - } -} - -function defineElements() { - customElements.define("ueb-color-handler", ColorHandlerElement); - ElementFactory.registerElement("ueb-color-handler", ColorHandlerElement); - customElements.define("ueb-input", InputElement); - ElementFactory.registerElement("ueb-input", InputElement); - customElements.define("ueb-link", LinkElement); - ElementFactory.registerElement("ueb-link", LinkElement); - customElements.define("ueb-node", NodeElement); - ElementFactory.registerElement("ueb-node", NodeElement); - customElements.define("ueb-pin", PinElement); - ElementFactory.registerElement("ueb-pin", PinElement); - customElements.define("ueb-selector", SelectorElement); - ElementFactory.registerElement("ueb-selector", SelectorElement); - customElements.define("ueb-ui-slider", ColorSliderElement); - ElementFactory.registerElement("ueb-ui-slider", ColorSliderElement); - customElements.define("ueb-window", WindowElement); - ElementFactory.registerElement("ueb-window", WindowElement); -} - +} + +/** @typedef {import("../Blueprint").BlueprintConstructor} BlueprintConstructor */ + +/** @extends {IFromToPositionedElement} */ +class SelectorElement extends IFromToPositionedElement { + + /** @type {FastSelectionModel} */ + selectionModel = null + + constructor() { + super(); + super.initialize({}, new SelectorTemplate()); + } + + static newObject() { + return new SelectorElement() + } + + initialize() { + // Initialized in the constructor, this method does nothing + } + + /** @param {Number[]} initialPosition */ + beginSelect(initialPosition) { + const blueprintConstructor = /** @type {BlueprintConstructor} */(this.blueprint.constructor); + this.blueprint.selecting = true; + this.setBothLocations(initialPosition); + this.selectionModel = new FastSelectionModel( + initialPosition, + this.blueprint.getNodes(), + blueprintConstructor.nodeBoundariesSupplier, + blueprintConstructor.nodeSelectToggleFunction + ); + } + + /** @param {Number[]} finalPosition */ + selectTo(finalPosition) { + this.selectionModel.selectTo(finalPosition); + this.toX = finalPosition[0]; + this.toY = finalPosition[1]; + } + + endSelect() { + this.blueprint.selecting = false; + this.selectionModel = null; + this.fromX = 0; + this.fromY = 0; + this.toX = 0; + this.toY = 0; + } +} + +/** @typedef {typeof WindowElement} WindowElementConstructor */ + +/** + * @template {WindowTemplate} T + * @extends {IDraggableElement} + */ +class WindowElement extends IDraggableElement { + + static #typeTemplateMap = { + "window": WindowTemplate, + "color-picker": ColorPickerWindowTemplate, + } + + static properties = { + ...IDraggableElement.properties, + type: { + type: WindowTemplate, + attribute: "data-type", + reflect: true, + converter: { + fromAttribute: (value, type) => WindowElement.#typeTemplateMap[value], + toAttribute: (value, type) => + Object.entries(WindowElement.#typeTemplateMap).find(([k, v]) => value.constructor === v)?.[0], + }, + }, + } + + static newObject(entity = {}, template = entity.type ?? new WindowTemplate()) { + const result = new WindowElement(); + result.initialize(entity, template); + return result + } + + initialize(entity = {}, template = entity.type ?? new WindowTemplate()) { + entity.windowOptions ??= {}; + this.type = entity.type; + this.windowOptions = entity.windowOptions; + super.initialize(entity, template); + } + + computeSizes() { + const bounding = this.getBoundingClientRect(); + this.sizeX = bounding.width; + this.sizeY = bounding.height; + } + + cleanup() { + super.cleanup(); + this.acknowledgeClose(); + } + + acknowledgeClose() { + let deleteEvent = new CustomEvent(Configuration.windowCloseEventName); + this.dispatchEvent(deleteEvent); + } +} + +function defineElements() { + customElements.define("ueb-color-handler", ColorHandlerElement); + ElementFactory.registerElement("ueb-color-handler", ColorHandlerElement); + customElements.define("ueb-input", InputElement); + ElementFactory.registerElement("ueb-input", InputElement); + customElements.define("ueb-link", LinkElement); + ElementFactory.registerElement("ueb-link", LinkElement); + customElements.define("ueb-node", NodeElement); + ElementFactory.registerElement("ueb-node", NodeElement); + customElements.define("ueb-pin", PinElement); + ElementFactory.registerElement("ueb-pin", PinElement); + customElements.define("ueb-selector", SelectorElement); + ElementFactory.registerElement("ueb-selector", SelectorElement); + customElements.define("ueb-ui-slider", ColorSliderElement); + ElementFactory.registerElement("ueb-ui-slider", ColorSliderElement); + customElements.define("ueb-window", WindowElement); + ElementFactory.registerElement("ueb-window", WindowElement); +} + /** * @typedef {import("../entity/IEntity").default} IEntity * @typedef {import("../entity/IEntity").AnyValue} AnyValue @@ -9790,8 +9830,8 @@ class GeneralSerializer extends ISerializer { let result = this.wrap(this.subWrite(entity, [], object, insideString), object); return result } -} - +} + /** * @typedef {import("../entity/IEntity").default} IEntity * @typedef {import("../entity/IEntity").AnyValue} AnyValue @@ -9824,8 +9864,8 @@ class CustomSerializer extends GeneralSerializer { let result = this.#objectWriter(object, insideString); return result } -} - +} + /** * @typedef {import("../entity/IEntity").AnyValue} AnyValue * @typedef {import("../entity/IEntity").AnyValueConstructor<*>} AnyValueConstructor @@ -9851,8 +9891,8 @@ class ToStringSerializer extends GeneralSerializer { ? `"${Utility.escapeString(object.toString())}"` // String will have quotes if not inside a string already : Utility.escapeString(object.toString()) } -} - +} + /** * @typedef {import("../entity/IEntity").AnySimpleValue} AnySimpleValue * @typedef {import("../entity/IEntity").AnyValue} AnyValue @@ -10078,9 +10118,9 @@ function initializeSerializerFactory() { VectorEntity, new GeneralSerializer(bracketsWrapped, VectorEntity) ); -} - +} + initializeSerializerFactory(); -defineElements(); - -export { Blueprint, Configuration, LinkElement, NodeElement, Utility }; +defineElements(); + +export { Blueprint, Configuration, LinkElement, NodeElement, Utility }; diff --git a/dist/ueblueprint.min.js b/dist/ueblueprint.min.js index a5e8a49..038348d 100644 --- a/dist/ueblueprint.min.js +++ b/dist/ueblueprint.min.js @@ -14,20 +14,20 @@ const e=window,t=e.ShadowRoot&&(void 0===e.ShadyCSS||e.ShadyCSS.nativeShadow)&&" * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ -var b;g.finalized=!0,g.elementProperties=new Map,g.elementStyles=[],g.shadowRootOptions={mode:"open"},null==h||h({ReactiveElement:g}),(null!==(o=l.reactiveElementVersions)&&void 0!==o?o:l.reactiveElementVersions=[]).push("1.4.2");const v=window,f=v.trustedTypes,y=f?f.createPolicy("lit-html",{createHTML:e=>e}):void 0,w=`lit$${(Math.random()+"").slice(9)}$`,E="?"+w,C=`<${E}>`,S=document,P=(e="")=>S.createComment(e),x=e=>null===e||"object"!=typeof e&&"function"!=typeof e,k=Array.isArray,L=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,N=/-->/g,T=/>/g,A=RegExp(">|[ \t\n\f\r](?:([^\\s\"'>=/]+)([ \t\n\f\r]*=[ \t\n\f\r]*(?:[^ \t\n\f\r\"'`<>=]|(\"|')|))|$)","g"),M=/'/g,B=/"/g,D=/^(?:script|style|textarea|title)$/i,$=(e=>(t,...i)=>({_$litType$:e,strings:t,values:i}))(1),H=Symbol.for("lit-noChange"),O=Symbol.for("lit-nothing"),V=new WeakMap,z=S.createTreeWalker(S,129,null,!1),I=(e,t)=>{const i=e.length-1,n=[];let s,r=2===t?"":"",a=L;for(let t=0;t"===l[0]?(a=null!=s?s:L,u=-1):void 0===l[1]?u=-2:(u=a.lastIndex-l[2].length,o=l[1],a=void 0===l[3]?A:'"'===l[3]?B:M):a===B||a===M?a=A:a===N||a===T?a=L:(a=A,s=void 0);const h=a===A&&e[t+1].startsWith("/>")?" ":"";r+=a===L?i+C:u>=0?(n.push(o),i.slice(0,u)+"$lit$"+i.slice(u)+w+h):i+w+(-2===u?(n.push(void 0),t):h)}const o=r+(e[i]||"")+(2===t?"":"");if(!Array.isArray(e)||!e.hasOwnProperty("raw"))throw Error("invalid template strings array");return[void 0!==y?y.createHTML(o):o,n]};class R{constructor({strings:e,_$litType$:t},i){let n;this.parts=[];let s=0,r=0;const a=e.length-1,o=this.parts,[l,u]=I(e,t);if(this.el=R.createElement(l,i),z.currentNode=this.el.content,2===t){const e=this.el.content,t=e.firstChild;t.remove(),e.append(...t.childNodes)}for(;null!==(n=z.nextNode())&&o.length0){n.textContent=f?f.emptyScript:"";for(let i=0;ik(e)||"function"==typeof(null==e?void 0:e[Symbol.iterator]))(e)?this.k(e):this.g(e)}O(e,t=this._$AB){return this._$AA.parentNode.insertBefore(e,t)}T(e){this._$AH!==e&&(this._$AR(),this._$AH=this.O(e))}g(e){this._$AH!==O&&x(this._$AH)?this._$AA.nextSibling.data=e:this.T(S.createTextNode(e)),this._$AH=e}$(e){var t;const{values:i,_$litType$:n}=e,s="number"==typeof n?this._$AC(e):(void 0===n.el&&(n.el=R.createElement(n.h,this.options)),n);if((null===(t=this._$AH)||void 0===t?void 0:t._$AD)===s)this._$AH.p(i);else{const e=new F(s,this),t=e.v(this.options);e.p(i),this.T(t),this._$AH=e}}_$AC(e){let t=V.get(e.strings);return void 0===t&&V.set(e.strings,t=new R(e)),t}k(e){k(this._$AH)||(this._$AH=[],this._$AR());const t=this._$AH;let i,n=0;for(const s of e)n===t.length?t.push(i=new G(this.O(P()),this.O(P()),this,this.options)):i=t[n],i._$AI(s),n++;n2||""!==i[0]||""!==i[1]?(this._$AH=Array(i.length-1).fill(new String),this.strings=i):this._$AH=O}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(e,t=this,i,n){const s=this.strings;let r=!1;if(void 0===s)e=_(this,e,t,0),r=!x(e)||e!==this._$AH&&e!==H,r&&(this._$AH=e);else{const n=e;let a,o;for(e=s[0],a=0;ae}):void 0,w=`lit$${(Math.random()+"").slice(9)}$`,E="?"+w,C=`<${E}>`,S=document,x=(e="")=>S.createComment(e),P=e=>null===e||"object"!=typeof e&&"function"!=typeof e,k=Array.isArray,L=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,N=/-->/g,T=/>/g,A=RegExp(">|[ \t\n\f\r](?:([^\\s\"'>=/]+)([ \t\n\f\r]*=[ \t\n\f\r]*(?:[^ \t\n\f\r\"'`<>=]|(\"|')|))|$)","g"),M=/'/g,B=/"/g,D=/^(?:script|style|textarea|title)$/i,$=(e=>(t,...i)=>({_$litType$:e,strings:t,values:i}))(1),H=Symbol.for("lit-noChange"),O=Symbol.for("lit-nothing"),V=new WeakMap,z=S.createTreeWalker(S,129,null,!1),I=(e,t)=>{const i=e.length-1,n=[];let s,r=2===t?"":"",a=L;for(let t=0;t"===l[0]?(a=null!=s?s:L,u=-1):void 0===l[1]?u=-2:(u=a.lastIndex-l[2].length,o=l[1],a=void 0===l[3]?A:'"'===l[3]?B:M):a===B||a===M?a=A:a===N||a===T?a=L:(a=A,s=void 0);const h=a===A&&e[t+1].startsWith("/>")?" ":"";r+=a===L?i+C:u>=0?(n.push(o),i.slice(0,u)+"$lit$"+i.slice(u)+w+h):i+w+(-2===u?(n.push(void 0),t):h)}const o=r+(e[i]||"")+(2===t?"":"");if(!Array.isArray(e)||!e.hasOwnProperty("raw"))throw Error("invalid template strings array");return[void 0!==y?y.createHTML(o):o,n]};class R{constructor({strings:e,_$litType$:t},i){let n;this.parts=[];let s=0,r=0;const a=e.length-1,o=this.parts,[l,u]=I(e,t);if(this.el=R.createElement(l,i),z.currentNode=this.el.content,2===t){const e=this.el.content,t=e.firstChild;t.remove(),e.append(...t.childNodes)}for(;null!==(n=z.nextNode())&&o.length0){n.textContent=f?f.emptyScript:"";for(let i=0;ik(e)||"function"==typeof(null==e?void 0:e[Symbol.iterator]))(e)?this.k(e):this.g(e)}O(e,t=this._$AB){return this._$AA.parentNode.insertBefore(e,t)}T(e){this._$AH!==e&&(this._$AR(),this._$AH=this.O(e))}g(e){this._$AH!==O&&P(this._$AH)?this._$AA.nextSibling.data=e:this.T(S.createTextNode(e)),this._$AH=e}$(e){var t;const{values:i,_$litType$:n}=e,s="number"==typeof n?this._$AC(e):(void 0===n.el&&(n.el=R.createElement(n.h,this.options)),n);if((null===(t=this._$AH)||void 0===t?void 0:t._$AD)===s)this._$AH.p(i);else{const e=new F(s,this),t=e.v(this.options);e.p(i),this.T(t),this._$AH=e}}_$AC(e){let t=V.get(e.strings);return void 0===t&&V.set(e.strings,t=new R(e)),t}k(e){k(this._$AH)||(this._$AH=[],this._$AR());const t=this._$AH;let i,n=0;for(const s of e)n===t.length?t.push(i=new G(this.O(x()),this.O(x()),this,this.options)):i=t[n],i._$AI(s),n++;n2||""!==i[0]||""!==i[1]?(this._$AH=Array(i.length-1).fill(new String),this.strings=i):this._$AH=O}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(e,t=this,i,n){const s=this.strings;let r=!1;if(void 0===s)e=_(this,e,t,0),r=!P(e)||e!==this._$AH&&e!==H,r&&(this._$AH=e);else{const n=e;let a,o;for(e=s[0],a=0;a{var n,s;const r=null!==(n=null==i?void 0:i.renderBefore)&&void 0!==n?n:t;let a=r._$litPart$;if(void 0===a){const e=null!==(s=null==i?void 0:i.renderBefore)&&void 0!==s?s:null;r._$litPart$=a=new G(t.insertBefore(P(),e),e,void 0,null!=i?i:{})}return a._$AI(e),a})(t,this.renderRoot,this.renderOptions)}connectedCallback(){var e;super.connectedCallback(),null===(e=this._$Do)||void 0===e||e.setConnected(!0)}disconnectedCallback(){var e;super.disconnectedCallback(),null===(e=this._$Do)||void 0===e||e.setConnected(!1)}render(){return H}}Q.finalized=!0,Q._$litElement$=!0,null===(q=globalThis.litElementHydrateSupport)||void 0===q||q.call(globalThis,{LitElement:Q});const ee=globalThis.litElementPolyfillSupport;null==ee||ee({LitElement:Q}),(null!==(J=globalThis.litElementVersions)&&void 0!==J?J:globalThis.litElementVersions=[]).push("3.2.2");class te{static nodeColors={blue:r``,gray:r``,green:r``,red:r``,turquoise:r``};static alphaPattern="repeating-conic-gradient(#7c8184 0% 25%, #c2c3c4 0% 50%) 50% / 10px 10px";static colorDragEventName="ueb-color-drag";static colorPickEventName="ueb-color-pick";static colorWindowEventName="ueb-color-window";static colorWindowName="Color Picker";static defaultCommentHeight=96;static defaultCommentWidth=400;static deleteNodesKeyboardKey="Delete";static distanceThreshold=5;static dragEventName="ueb-drag";static dragGeneralEventName="ueb-drag-general";static edgeScrollThreshold=50;static editTextEventName={begin:"ueb-edit-text-begin",end:"ueb-edit-text-end"};static enableZoomIn=["LeftControl","RightControl"];static expandGridSize=400;static focusEventName={begin:"blueprint-focus",end:"blueprint-unfocus"};static fontSize=r``;static gridAxisLineColor=r``;static gridExpandThreshold=.25;static gridLineColor=r``;static gridLineWidth=1;static gridSet=8;static gridSetLineColor=r``;static gridShrinkThreshold=4;static gridSize=16;static hexColorRegex=/^\s*#(?[0-9a-fA-F]{2})(?[0-9a-fA-F]{2})(?[0-9a-fA-F]{2})([0-9a-fA-F]{2})?|#(?[0-9a-fA-F])(?[0-9a-fA-F])(?[0-9a-fA-F])\s*$/;static keysSeparator="+";static knotOffset=[-26,-16];static linkCurveHeight=15;static linkCurveWidth=80;static linkMinWidth=100;static linkRightSVGPath=(e,t,i)=>{let n=100-e;return`M ${e} 0 C ${t.toFixed(3)} 0, ${i.toFixed(3)} 0, 50 50 S ${(n-t+e).toFixed(3)} 100, ${n.toFixed(3)} 100`};static maxZoom=7;static minZoom=-12;static mouseWheelFactor=.2;static nodeDragGeneralEventName="ueb-node-drag-general";static nodeDragEventName="ueb-node-drag";static nodeName=(e,t)=>`${e}_${t}`;static nodeRadius=8;static nodeReflowEventName="ueb-node-reflow";static nodeType={callArrayFunction:"/Script/BlueprintGraph.K2Node_CallArrayFunction",callFunction:"/Script/BlueprintGraph.K2Node_CallFunction",comment:"/Script/UnrealEd.EdGraphNode_Comment",commutativeAssociativeBinaryOperator:"/Script/BlueprintGraph.K2Node_CommutativeAssociativeBinaryOperator",componentBoundEvent:"/Script/BlueprintGraph.K2Node_ComponentBoundEvent",customEvent:"/Script/BlueprintGraph.K2Node_CustomEvent",doN:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:Do N",dynamicCast:"/Script/BlueprintGraph.K2Node_DynamicCast",enum:"/Script/CoreUObject.Enum",enumLiteral:"/Script/BlueprintGraph.K2Node_EnumLiteral",event:"/Script/BlueprintGraph.K2Node_Event",executionSequence:"/Script/BlueprintGraph.K2Node_ExecutionSequence",forEachElementInEnum:"/Script/BlueprintGraph.K2Node_ForEachElementInEnum",forEachLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForEachLoop",forEachLoopWithBreak:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForEachLoopWithBreak",forLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForLoop",forLoopWithBreak:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForLoopWithBreak",functionEntry:"/Script/BlueprintGraph.K2Node_FunctionEntry",getInputAxisKeyValue:"/Script/BlueprintGraph.K2Node_GetInputAxisKeyValue",ifThenElse:"/Script/BlueprintGraph.K2Node_IfThenElse",inputAxisKeyEvent:"/Script/BlueprintGraph.K2Node_InputAxisKeyEvent",inputDebugKey:"/Script/InputBlueprintNodes.K2Node_InputDebugKey",inputKey:"/Script/BlueprintGraph.K2Node_InputKey",isValid:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:IsValid",knot:"/Script/BlueprintGraph.K2Node_Knot",macro:"/Script/BlueprintGraph.K2Node_MacroInstance",makeArray:"/Script/BlueprintGraph.K2Node_MakeArray",makeMap:"/Script/BlueprintGraph.K2Node_MakeMap",makeSet:"/Script/BlueprintGraph.K2Node_MakeSet",pawn:"/Script/Engine.Pawn",promotableOperator:"/Script/BlueprintGraph.K2Node_PromotableOperator",reverseForEachLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ReverseForEachLoop",select:"/Script/BlueprintGraph.K2Node_Select",userDefinedEnum:"/Script/Engine.UserDefinedEnum",variableGet:"/Script/BlueprintGraph.K2Node_VariableGet",variableSet:"/Script/BlueprintGraph.K2Node_VariableSet",whileLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:WhileLoop"};static pinColor={"/Script/CoreUObject.Rotator":r``,"/Script/CoreUObject.Transform":r``,"/Script/CoreUObject.Vector":r``,bool:r``,byte:r``,class:r``,default:r``,delegate:r``,enum:r``,exec:r``,int:r``,int64:r``,interface:r``,name:r``,object:r``,real:r``,string:r``,struct:r``,text:r``,wildcard:r``};static pinInputWrapWidth=134;static removeEventName="ueb-element-delete";static scale={[-12]:.133333,[-11]:.166666,[-10]:.2,[-9]:.233333,[-8]:.266666,[-7]:.3,[-6]:.333333,[-5]:.375,[-4]:.5,[-3]:.675,[-2]:.75,[-1]:.875,0:1,1:1.25,2:1.375,3:1.5,4:1.675,5:1.75,6:1.875,7:2};static selectAllKeyboardKey="(bCtrl=True,Key=A)";static smoothScrollTime=1e3;static trackingMouseEventName={begin:"ueb-tracking-mouse-begin",end:"ueb-tracking-mouse-end"};static windowApplyEventName="ueb-window-apply";static windowApplyButtonText="OK";static windowCancelEventName="ueb-window-cancel";static windowCancelButtonText="Cancel";static windowCloseEventName="ueb-window-close";static ModifierKeys=["Ctrl","Shift","Alt","Meta"];static Keys={Backspace:"Backspace",Tab:"Tab",LeftControl:"ControlLeft",RightControl:"ControlRight",LeftShift:"ShiftLeft",RightShift:"ShiftRight",LeftAlt:"AltLeft",RightAlt:"AltRight",Enter:"Enter",Pause:"Pause",CapsLock:"CapsLock",Escape:"Escape",Space:"Space",PageUp:"PageUp",PageDown:"PageDown",End:"End",Home:"Home",ArrowLeft:"Left",ArrowUp:"Up",ArrowRight:"Right",ArrowDown:"Down",PrintScreen:"PrintScreen",Insert:"Insert",Delete:"Delete",Zero:"Digit0",One:"Digit1",Two:"Digit2",Three:"Digit3",Four:"Digit4",Five:"Digit5",Six:"Digit6",Seven:"Digit7",Eight:"Digit8",Nine:"Digit9",A:"KeyA",B:"KeyB",C:"KeyC",D:"KeyD",E:"KeyE",F:"KeyF",G:"KeyG",H:"KeyH",I:"KeyI",K:"KeyK",L:"KeyL",M:"KeyM",N:"KeyN",O:"KeyO",P:"KeyP",Q:"KeyQ",R:"KeyR",S:"KeyS",T:"KeyT",U:"KeyU",V:"KeyV",W:"KeyW",X:"KeyX",Y:"KeyY",Z:"KeyZ",NumPadZero:"Numpad0",NumPadOne:"Numpad1",NumPadTwo:"Numpad2",NumPadThree:"Numpad3",NumPadFour:"Numpad4",NumPadFive:"Numpad5",NumPadSix:"Numpad6",NumPadSeven:"Numpad7",NumPadEight:"Numpad8",NumPadNine:"Numpad9",Multiply:"NumpadMultiply",Add:"NumpadAdd",Subtract:"NumpadSubtract",Decimal:"NumpadDecimal",Divide:"NumpadDivide",F1:"F1",F2:"F2",F3:"F3",F4:"F4",F5:"F5",F6:"F6",F7:"F7",F8:"F8",F9:"F9",F10:"F10",F11:"F11",F12:"F12",NumLock:"NumLock",ScrollLock:"ScrollLock"}}class ie{#e;get target(){return this.#e}#t;get blueprint(){return this.#t}options;listenHandler=()=>this.listenEvents();unlistenHandler=()=>this.unlistenEvents();constructor(e,t,i={}){i.consumeEvent??=!1,i.listenOnFocus??=!1,i.unlistenOnTextEdit??=!1,this.#e=e,this.#t=t,this.options=i}setup(){this.options.listenOnFocus&&(this.blueprint.addEventListener(te.focusEventName.begin,this.listenHandler),this.blueprint.addEventListener(te.focusEventName.end,this.unlistenHandler)),this.options.unlistenOnTextEdit&&(this.blueprint.addEventListener(te.editTextEventName.begin,this.unlistenHandler),this.blueprint.addEventListener(te.editTextEventName.end,this.listenHandler)),this.blueprint.focused&&this.listenEvents()}cleanup(){this.unlistenEvents(),this.blueprint.removeEventListener(te.focusEventName.begin,this.listenHandler),this.blueprint.removeEventListener(te.focusEventName.end,this.unlistenHandler),this.blueprint.removeEventListener(te.editTextEventName.begin,this.unlistenHandler),this.blueprint.removeEventListener(te.editTextEventName.end,this.listenHandler)}listenEvents(){}unlistenEvents(){}}class ne{static#i=new Map;static registerSerializer(e,t){ne.#i.set(e,t)}static getSerializer(e){return ne.#i.get(e)}}class se{constructor(e){this.attributes=e}}class re{#n;get types(){return this.#n}constructor(...e){this.#n=e}getFirstType(){return this.#n[0]}}class ae{static emptyObj={};static booleanConverter={fromAttribute:(e,t)=>{},toAttribute:(e,t)=>!0===e?"true":!1===e?"false":""};static sigmoid(e,t=1.7){return 1/(1+e/(1-e)**-t)}static sigmoidPositive(e,t=3.7,i=1.1){return 1-Math.exp(-((e/i)**t))}static clamp(e,t=-1/0,i=1/0){return Math.min(Math.max(e,t),i)}static getScale(e){const t=e.blueprint?e.blueprint.getScale():getComputedStyle(e).getPropertyValue("--ueb-scale");return""!=t?parseFloat(t):1}static minDecimals(e,t=1,i=1e-8){const n=e*10**t;return Math.abs(n%1)>i?e.toString():e.toFixed(t)}static numberFromText(e=""){switch(e=e.toLowerCase()){case"zero":return 0;case"one":return 1;case"two":return 2;case"three":return 3;case"four":return 4;case"five":return 5;case"six":return 6;case"seven":return 7;case"eight":return 8;case"nine":return 9}}static roundDecimals(e,t=1){const i=10**t;return Math.round(e*i)/i}static approximatelyEqual(e,t,i=1e-8){return!(Math.abs(e-t)>i)}static convertLocation(e,t,i=!1){const n=i?1:1/ae.getScale(t),s=t.getBoundingClientRect();return[Math.round((e[0]-s.x)*n),Math.round((e[1]-s.y)*n)]}static isSerialized(e,t,i=ae.objectGet(e.constructor.attributes,t)){return i?.constructor===Object&&i.serialized}static objectGet(e,t,i){if(void 0!==e){if(!(t instanceof Array))throw new TypeError("UEBlueprint: Expected keys to be an array");return e instanceof se&&(e=e.attributes),0!=t.length&&t[0]in e&&void 0!==e[t[0]]?1==t.length?e[t[0]]:ae.objectGet(e[t[0]],t.slice(1),i):i}}static objectSet(e,t,i,n=!1,s=Object){if(!(t instanceof Array))throw new TypeError("Expected keys to be an array.");if(1==t.length){if(n||t[0]in e||void 0===e[t[0]])return e[t[0]]=i,!0}else if(t.length>0)return!n||e[t[0]]instanceof Object||(e[t[0]]=new s),ae.objectSet(e[t[0]],t.slice(1),i,n,s);return!1}static equals(e,t){return e?.equals&&t?.equals?e.equals(t):(e=ae.sanitize(e),t=ae.sanitize(t),e?.constructor===BigInt&&t?.constructor===Number?t=BigInt(t):e?.constructor===Number&&t?.constructor===BigInt&&(e=BigInt(e)),e===t||e instanceof Array&&t instanceof Array&&(e.length===t.length&&e.every(((e,i)=>ae.equals(e,t[i])))))}static getType(e){return null===e?null:e?.constructor===Object&&e?.type instanceof Function?e.type:e?.constructor}static isValueOfType(e,t,i=!1){return i&&null===e||e instanceof t||e?.constructor===t}static sanitize(e,t=e?.constructor){if(t instanceof re){let i=t.types.find((t=>ae.isValueOfType(e,t,!1)));i||(i=t.getFirstType()),t=i}return t&&!ae.isValueOfType(e,t,!0)&&(e=t===BigInt?BigInt(e):new t(e)),(e instanceof Boolean||e instanceof Number||e instanceof String||e instanceof BigInt)&&(e=e.valueOf()),e}static snapToGrid(e,t,i){return 1===i?[e,t]:[i*Math.round(e/i),i*Math.round(t/i)]}static mergeArrays(e=[],t=[]){let i=[];e=[...e],t=[...t];e:for(;;){for(let n=0;n","\n").replaceAll(/(\)/g,"")}static capitalFirstLetter(e){return 0===e.length?e:e.charAt(0).toUpperCase()+e.slice(1)}static formatStringName(e){return e.replace(/^\s*b/,"").replaceAll(/^K2(?:Node|node)?_|(?<=[a-z])(?=[A-Z0-9])|(?<=[A-Z])(?=[A-Z][a-z]|[0-9])|(?<=[014-9]|(?:2|3)(?!D(?:[^a-z]|$)))(?=[a-zA-Z])|\s*_+\s*|\s{2,}/g," ").split(" ").map((e=>ae.capitalFirstLetter(e))).join(" ").trim()}static getIdFromReference(e){return e.replace(/(?:.+\.)?([^\.]+)$/,"$1").replaceAll(/(?<=[a-z\d])(?=[A-Z])|(?<=[a-zA-Z])(?=\d)|(?<=[A-Z]{2})(?=[A-Z][a-z])/g,"-").toLowerCase()}static printLinearColor(e){return`${Math.round(255*e.R.valueOf())}, ${Math.round(255*e.G.valueOf())}, ${Math.round(255*e.B.valueOf())}`}static getPolarCoordinates(e,t,i=!1){let n=Math.atan2(t,e);return i&&n<0&&(n=2*Math.PI+n),[Math.sqrt(e*e+t*t),n]}static getCartesianCoordinates(e,t){return[e*Math.cos(t),e*Math.sin(t)]}static range(e=0,t=0,i=(t>=e?1:-1)){return Array.from({length:Math.ceil((t-e)/i)},((t,n)=>e+n*i))}static paste(e,t){const i=new ClipboardEvent("paste",{bubbles:!0,cancelable:!0,clipboardData:new DataTransfer});i.clipboardData.setData("text",t),e.dispatchEvent(i)}static animate(e,t,i,n,s=(e=>{const t=e**3.5;return t/(t+(1-e)**3.5)})){let r;const a=o=>{void 0===r&&(r=o);let l=(o-r)/i;ae.approximatelyEqual(l,1)||l>1?l=1:requestAnimationFrame(a);const u=e+(t-e)*s(l);n(u)};requestAnimationFrame(a)}}class oe{static attributes={};static defaultAttribute={showDefault:!0,nullable:!1,ignored:!1,serialized:!1,expected:!1};constructor(e={},t=!1){const i=(e,n,s={},r="")=>{const a=Object.keys(s),o=Object.keys(n),l=ae.mergeArrays(o,a);for(let o of l){let l=ae.objectGet(s,[o]),u=n[o];if(u instanceof se){e[o]={},i(e[o],u.attributes,s[o],o+".");continue}if(t||(o in n?a.length>0&&!(o in s)&&u.showDefault&&!u.ignored&&console.warn(`UEBlueprint: ${this.constructor.name} will add attribute ${r}${o} not defined in the serialized data`):console.warn(`UEBlueprint: Attribute ${r}${o} in the serialized data is not defined in ${this.constructor.name}.attributes`)),!u){e[o]=l;continue}let c=u.value,h=u.type;u.serialized&&h instanceof Function&&(h=h(this)),h instanceof Array&&(h=Array),c instanceof Function&&(c=c(this)),void 0===h&&(h=ae.getType(c));const d=u.predicate?t=>{Object.defineProperties(e,{["#"+o]:{writable:!0,enumerable:!1},[o]:{enumerable:!0,get(){return this["#"+o]},set(e){u.predicate?.(e)?this["#"+o]=e:console.warn(`UEBlueprint: Tried to assign attribute ${r}${o} to ${this.constructor.name} not satisfying the predicate`)}}}),this[o]=t}:t=>e[o]=t;void 0===l?(h instanceof re&&(h=null!=c?h.types.find((e=>c instanceof e||c.constructor==e))??h.getFirstType():h.getFirstType()),void 0===c&&(c=ae.sanitize(new h)),u.showDefault?(u.serialized&&h!==String&&c.constructor===String&&(c=ne.getSerializer(h).deserialize(c)),d(ae.sanitize(c,h))):d(void 0)):(l?.constructor===String&&u.serialized&&h!==String&&(l=ne.getSerializer(h).deserialize(l)),d(ae.sanitize(l,h)))}},n=this.constructor.attributes;e.constructor!==Object&&1===Object.keys(n).length&&(e={[Object.keys(n)[0]]:e}),i(this,n,e)}static cleanupAttributes(e,t=""){for(const i in e){if(e[i]instanceof se){this.cleanupAttributes(e[i].attributes,t+"."+i);continue}e[i].constructor!==Object&&(e[i]={value:e[i]});const n=e[i];if(void 0!==n.type||n.value instanceof Function||(n.type=ae.getType(n.value)),e[i]={...oe.defaultAttribute,...n},void 0===n.value&&void 0===n.type)throw new Error(`UEBlueprint: Expected either "type" or "value" property in ${this.name} attribute ${t}`+i);null===n.value&&(e[i].nullable=!0)}}static isValueOfType(e,t){return null!=e&&(e instanceof t||e.constructor===t)}static expectsAllKeys(){return!Object.values(this.attributes).filter((e=>!e.ignored)).some((e=>!e.expected))}unexpectedKeys(){return Object.keys(this).length-Object.keys(this.constructor.attributes).length}equals(e){const t=Object.keys(this),i=Object.keys(this);if(t.length!=i.length)return!1;for(const i of t){if(this[i]instanceof oe&&!this[i].equals(e[i]))return!1;if(!ae.equals(this[i],e[i]))return!1}return!0}}class le extends oe{static attributes={...super.attributes,value:{value:0,predicate:e=>e%1==0&&e>1<<31&&e<-(1<<31)}};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e),this.value}valueOf(){return this.value}toString(){return this.value.toString()}}class ue extends le{static attributes={...super.attributes,value:{...super.attributes.value,predicate:e=>e%1==0&&e>=0&&e<256}};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e)}}class ce extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class he extends ce{}class de extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}static generateGuid(e=!0){let t=new Uint32Array(4);!0===e&&crypto.getRandomValues(t);let i="";return t.forEach((e=>{i+=("0".repeat(8)+e.toString(16).toUpperCase()).slice(-8)})),new de({value:i})}constructor(e){e||(e=de.generateGuid().value),super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class pe extends oe{static attributes={type:"",path:""};static{this.cleanupAttributes(this.attributes)}constructor(e={}){e.constructor===String&&(e={path:e}),super(e),this.type,this.path}getName(){return this.path.match(/[^\.\/]+$/)?.[0]??""}}class me extends oe{static attributes={MemberParent:{type:pe,showDefault:!1},MemberName:{type:String,showDefault:!1},MemberGuid:{type:de,showDefault:!1}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.MemberParent,this.MemberName}}class ge extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}static attributeConverter={fromAttribute:(e,t)=>new ge(e),toAttribute:(e,t)=>e.toString()};constructor(e){super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class be extends oe{static attributes={...super.attributes,value:{value:0n,predicate:e=>e>=-(1n<<63n)&&e<1n<<63n}};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e),this.value}valueOf(){return this.value}toString(){return this.value.toString()}}class ve extends oe{static lookbehind="INVTEXT";static attributes={value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}}class fe extends oe{static attributes={ActionName:"",bShift:!1,bCtrl:!1,bAlt:!1,bCmd:!1,Key:{type:ge}};static{this.cleanupAttributes(this.attributes)}constructor(e={}){e.ActionName=e.ActionName??"",e.bShift=e.bShift??!1,e.bCtrl=e.bCtrl??!1,e.bAlt=e.bAlt??!1,e.bCmd=e.bCmd??!1,super(e),this.ActionName,this.bShift,this.bCtrl,this.bAlt,this.bCmd,this.Key}}class ye extends oe{static attributes={value:0};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e),this.value=ae.clamp(this.value,0,1)}valueOf(){return this.value}toString(){return this.value.toFixed(6)}}class we extends oe{static attributes={R:{type:ye,expected:!0},G:{type:ye,expected:!0},B:{type:ye,expected:!0},A:{type:ye,value:()=>new ye(1)},H:{type:ye,showDefault:!0,ignored:!0},S:{type:ye,showDefault:!0,ignored:!0},V:{type:ye,showDefault:!0,ignored:!0}};static{this.cleanupAttributes(this.attributes)}static linearToSRGB(e){return e<=0?0:e>=1?1:e<.0031308?12.92*e:1.055*Math.pow(e,1/2.4)-.055}static sRGBtoLinear(e){return e<=0?0:e>=1?1:e<.04045?e/12.92:Math.pow((e+.055)/1.055,2.4)}static getWhite(){return new we({R:1,G:1,B:1})}constructor(e){e instanceof Array&&(e={R:e[0]??0,G:e[1]??0,B:e[2]??0,A:e[3]??1}),super(e),this.R,this.G,this.B,this.A,this.H,this.S,this.V,this.#s()}#s(){const e=this.R.value,t=this.G.value,i=this.B.value;if(ae.approximatelyEqual(e,t)&&ae.approximatelyEqual(e,i)&&ae.approximatelyEqual(t,i))return this.S.value=0,void(this.V.value=e);const n=Math.max(e,t,i),s=Math.min(e,t,i),r=n-s;let a;switch(n){case s:a=0;break;case e:a=(t-i)/r+(te.toString(16).toUpperCase().padStart(2,"0"))).join("")}toSRGBAString(){return this.toSRGBA().map((e=>e.toString(16).toUpperCase().padStart(2,"0"))).join("")}toHSVA(){return[this.H.value,this.S.value,this.V.value,this.A.value]}toNumber(){return(Math.round(255*this.R.value)<<24)+(Math.round(255*this.G.value)<<16)+(Math.round(255*this.B.value)<<8)+Math.round(255*this.A.value)}setFromRGBANumber(e){this.A.value=(255&e)/255,this.B.value=(e>>8&255)/255,this.G.value=(e>>16&255)/255,this.R.value=(e>>24&255)/255,this.#s()}setFromSRGBANumber(e){this.A.value=(255&e)/255,this.B.value=we.sRGBtoLinear((e>>8&255)/255),this.G.value=we.sRGBtoLinear((e>>16&255)/255),this.R.value=we.sRGBtoLinear((e>>24&255)/255),this.#s()}toString(){return ae.printLinearColor(this)}}class Ee extends oe{static lookbehind="NSLOCTEXT";static attributes={namespace:"",key:"",value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.namespace,this.key,this.value}toString(){return ae.capitalFirstLetter(this.value)}}class Ce extends oe{static attributes={MacroGraph:{type:pe},GraphBlueprint:{type:pe},GraphGuid:{type:de}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.MacroGraph,this.GraphBlueprint,this.GuidEntity}getMacroName(){const e=this.MacroGraph.path.search(":");return this.MacroGraph.path.substring(e+1)}}class Se extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class Pe extends oe{static attributes={objectName:{type:Se},pinGuid:{type:de}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.objectName,this.pinGuid}}class xe extends oe{static attributes={TerminalCategory:{value:"",showDefault:!1},TerminalSubCategory:{value:"",showDefault:!1},bTerminalIsConst:{value:!1,showDefault:!1},bTerminalIsWeakPointer:{value:!1,showDefault:!1},bTerminalIsUObjectWrapper:{value:!1,showDefault:!1}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.TerminalCategory,this.TerminalSubCategory,this.bTerminalIsConst,this.bTerminalIsWeakPointer,this.bTerminalIsUObjectWrapper}}class ke extends oe{static attributes={R:{value:0},P:{value:0},Y:{value:0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.R,this.P,this.Y}getRoll(){return this.R}getPitch(){return this.P}getYaw(){return this.Y}}class Le extends ke{}class Ne extends oe{static attributes={X:{value:0,expected:!0},Y:{value:0,expected:!0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.X,this.Y}}class Te extends Ne{}class Ae extends oe{static attributes={X:{value:0,expected:!0},Y:{value:0,expected:!0},Z:{value:0,expected:!0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.X,this.Y,this.Z}}class Me extends Ae{}class Be extends oe{static#r={"/Script/CoreUObject.LinearColor":we,"/Script/CoreUObject.Rotator":ke,"/Script/CoreUObject.Vector":Ae,"/Script/CoreUObject.Vector2D":Ne,bool:Boolean,byte:ue,enum:he,exec:String,int:le,int64:be,name:String,real:Number,string:String};static#a={"/Script/CoreUObject.Vector2D":Te,"/Script/CoreUObject.Vector":Me,"/Script/CoreUObject.Rotator":Le};static lookbehind="Pin";static attributes={PinId:{type:de},PinName:"",PinFriendlyName:{type:new re(Ee,String),showDefault:!1},PinToolTip:{type:String,showDefault:!1},Direction:{type:String,showDefault:!1},PinType:new se({PinCategory:"",PinSubCategory:"",PinSubCategoryObject:{type:pe},PinSubCategoryMemberReference:{type:me,value:null},PinValueType:{type:xe,value:null},ContainerType:{type:Se},bIsReference:!1,bIsConst:!1,bIsWeakPointer:!1,bIsUObjectWrapper:!1,bSerializeAsSinglePrecisionFloat:!1}),LinkedTo:{type:[Pe],showDefault:!1},DefaultValue:{type:e=>e.getEntityType(!0)??String,serialized:!0,showDefault:!1},AutogeneratedDefaultValue:{type:String,showDefault:!1},DefaultObject:{type:pe,showDefault:!1,value:null},PersistentGuid:{type:de},bHidden:!1,bNotConnectable:!1,bDefaultValueIsReadOnly:!1,bDefaultValueIsIgnored:!1,bAdvancedView:!1,bOrphanedPin:!1};static{this.cleanupAttributes(this.attributes)}constructor(e={},t=!1){super(e,t),this.PinId,this.PinName,this.PinFriendlyName,this.PinToolTip,this.Direction,this.PinType,this.LinkedTo,this.DefaultValue,this.AutogeneratedDefaultValue,this.DefaultObject,this.PersistentGuid,this.bHidden,this.bNotConnectable,this.bDefaultValueIsReadOnly,this.bDefaultValueIsIgnored,this.bAdvancedView,this.bOrphanedPin}getType(){const e=this.PinType.PinSubCategoryObject;return"struct"===this.PinType.PinCategory||"object"===this.PinType.PinCategory?e.path:"byte"!==this.PinType.PinCategory||e.type!==te.nodeType.enum&&e.type!==te.nodeType.userDefinedEnum?this.PinType.PinCategory:"enum"}getEntityType(e=!1){const t=this.getType(),i=Be.#r[t],n=Be.#a[t];return e&&void 0!==n?n:i}getDisplayName(){let e=null;return this.PinToolTip&&(e=this.PinToolTip.match(/\s*(.+?(?=\n)|.+\S)\s*/))?ae.formatStringName(e[1]):ae.formatStringName(this.PinName)}copyTypeFrom(e){this.PinType.PinCategory=e.PinType.PinCategory,this.PinType.PinSubCategory=e.PinType.PinSubCategory,this.PinType.PinSubCategoryObject=e.PinType.PinSubCategoryObject,this.PinType.PinSubCategoryMemberReference=e.PinType.PinSubCategoryMemberReference,this.PinType.PinValueType=e.PinType.PinValueType,this.PinType.ContainerType=e.PinType.ContainerType,this.PinType.bIsReference=e.PinType.bIsReference,this.PinType.bIsConst=e.PinType.bIsConst,this.PinType.bIsWeakPointer=e.PinType.bIsWeakPointer,this.PinType.bIsUObjectWrapper=e.PinType.bIsUObjectWrapper,this.PinType.bSerializeAsSinglePrecisionFloat=e.PinType.bSerializeAsSinglePrecisionFloat}getDefaultValue(e=!1){return void 0===this.DefaultValue&&e&&(this.DefaultValue=new(this.getEntityType(!0))),this.DefaultValue}isExecution(){return"exec"===this.PinType.PinCategory}isHidden(){return this.bHidden}isInput(){return!this.bHidden&&"EGPD_Output"!=this.Direction}isOutput(){return!this.bHidden&&"EGPD_Output"==this.Direction}isLinked(){return this.LinkedTo?.length>0??!1}linkTo(e,t){const i=this.LinkedTo?.some((i=>i.objectName.toString()==e&&i.pinGuid.valueOf()==t.PinId.valueOf()));return!i&&((this.LinkedTo??=[]).push(new Pe({objectName:e,pinGuid:t.PinId})),!0)}unlinkFrom(e,t){const i=this.LinkedTo?.findIndex((i=>i.objectName.toString()==e&&i.pinGuid.valueOf()==t.PinId.valueOf()));return i>=0&&(this.LinkedTo.splice(i,1),0!==this.LinkedTo.length||Be.attributes.LinkedTo.showDefault||(this.LinkedTo=void 0),!0)}getSubCategory(){return this.PinType.PinSubCategoryObject.path}pinColor(){return te.pinColor[this.getType()]??te.pinColor[this.PinType.PinCategory]??te.pinColor.default}}class De{static array=$``;static branchNode=$``;static breakStruct=$``;static cast=$``;static close=$``;static correct=$``;static delegate=$``;static doN=$``;static enum=$``;static event=$``;static execPin=$``;static expandIcon=$``;static forEachLoop=$``;static functionSymbol=$``;static gamepad=$``;static genericPin=$``;static keyboard=$``;static loop=$``;static macro=$``;static map=$``;static makeArray=$``;static makeMap=$``;static makeSet=$``;static makeStruct=$``;static mouse=$``;static questionMark=$``;static referencePin=$``;static reject=$``;static set=$``;static select=$``;static sequence=$``;static touchpad=$``}class $e extends oe{static attributes={MemberScope:{value:"",showDefault:!1},MemberName:"",MemberGuid:{type:de},bSelfContext:{value:!1,showDefault:!1}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.MemberName,this.GuidEntity,this.bSelfContext}}class He extends oe{static attributes={Class:{type:pe},Name:"",AxisKey:{type:ce,showDefault:!1},InputAxisKey:{type:ce,showDefault:!1},bIsPureFunc:{value:!1,showDefault:!1},bIsConstFunc:{value:!1,showDefault:!1},VariableReference:{type:$e,value:null,showDefault:!1},SelfContextInfo:{type:ce,value:null,showDefault:!1},DelegatePropertyName:{type:String,showDefault:!1},DelegateOwnerClass:{type:pe,showDefault:!1},ComponentPropertyName:{type:String,showDefault:!1},EventReference:{type:me,value:null,showDefault:!1},FunctionReference:{type:me,value:null,showDefault:!1},CustomFunctionName:{type:String,showDefault:!1},TargetType:{type:pe,value:null,showDefault:!1},MacroGraphReference:{type:Ce,value:null,showDefault:!1},Enum:{type:pe,showDefault:!1},InputKey:{type:ce,showDefault:!1},bOverrideFunction:{type:Boolean,showDefault:!1},bInternalEvent:{type:Boolean,showDefault:!1},bConsumeInput:{type:Boolean,showDefault:!1},bExecuteWhenPaused:{type:Boolean,showDefault:!1},bOverrideParentBinding:{type:Boolean,showDefault:!1},bControl:{type:Boolean,showDefault:!1},bAlt:{type:Boolean,showDefault:!1},bShift:{type:Boolean,showDefault:!1},bCommand:{type:Boolean,showDefault:!1},CommentColor:{type:we,showDefault:!1},bCommentBubbleVisible_InDetailsPanel:{type:Boolean,showDefault:!1},bColorCommentBubble:{type:Boolean,value:!1,showDefault:!1},MoveMode:{type:ce,showDefault:!1},NodePosX:{type:le,showDefault:!1},NodePosY:{type:le,showDefault:!1},NodeWidth:{type:le,showDefault:!1},NodeHeight:{type:le,showDefault:!1},bCommentBubblePinned:{type:Boolean,showDefault:!1},bCommentBubbleVisible:{type:Boolean,showDefault:!1},NodeComment:{type:String,showDefault:!1},AdvancedPinDisplay:{type:ge,value:null,showDefault:!1},EnabledState:{type:ge,value:null,showDefault:!1},NodeGuid:{type:de},ErrorType:{type:le,showDefault:!1},ErrorMsg:{type:String,value:"",showDefault:!1},CustomProperties:{type:[Be]}};static nameRegex=/^(\w+?)(?:_(\d+))?$/;static sequencerScriptingNameRegex=/\/Script\/SequencerScripting\.MovieSceneScripting(.+)Channel/;static#o={A_AccentGrave:"à",Add:"Num +",C_Cedille:"ç",Decimal:"Num .",Divide:"Num /",E_AccentAigu:"é",E_AccentGrave:"è",F1:"F1",F10:"F10",F11:"F11",F12:"F12",F2:"F2",F3:"F3",F4:"F4",F5:"F5",F6:"F6",F7:"F7",F8:"F8",F9:"F9",Gamepad_Special_Left_X:"Touchpad Button X Axis",Gamepad_Special_Left_Y:"Touchpad Button Y Axis",Mouse2D:"Mouse XY 2D-Axis",Multiply:"Num *",Section:"§",Subtract:"Num -",Tilde:"`"};static{this.cleanupAttributes(this.attributes)}static keyName(e){let t=He.#o[e];if(t)return t;if(t=ae.numberFromText(e)?.toString(),t)return t;const i=e.match(/NumPad([a-zA-Z]+)/);return i&&(t=ae.numberFromText(i[1]),t)?"Num "+t:void 0}constructor(e,t=!1){super(e,t),this.Class,this.Name,this.AxisKey,this.InputAxisKey,this.bIsPureFunc,this.bIsConstFunc,this.VariableReference,this.SelfContextInfo,this.DelegatePropertyName,this.DelegateOwnerClass,this.ComponentPropertyName,this.EventReference,this.FunctionReference,this.CustomFunctionName,this.TargetType,this.MacroGraphReference,this.Enum,this.InputKey,this.bOverrideFunction,this.bInternalEvent,this.bConsumeInput,this.bExecuteWhenPaused,this.bOverrideParentBinding,this.bControl,this.bAlt,this.bShift,this.bCommand,this.CommentColor,this.bCommentBubbleVisible_InDetailsPanel,this.NodePosX,this.NodePosY,this.NodeWidth,this.NodeHeight,this.bCommentBubblePinned,this.bCommentBubbleVisible,this.NodeComment,this.AdvancedPinDisplay,this.EnabledState,this.NodeGuid,this.ErrorType,this.ErrorMsg,this.CustomProperties}getClass(){return this.Class.path}getType(){let e=this.getClass();return e===te.nodeType.macro?this.MacroGraphReference.MacroGraph.path:e}getObjectName(e=!1){return e?this.getNameAndCounter()[0]:this.Name}getNameAndCounter(){const e=this.getObjectName(!1).match(He.nameRegex);let t="",i=null;return e?(e.length>1&&(t=e[1]),e.length>2&&(i=parseInt(e[2])),[t,i]):["",0]}getCounter(){return this.getNameAndCounter()[1]}getNodeWidth(){return this.NodeWidth??this.getType()==te.nodeType.comment?te.defaultCommentWidth:void 0}setNodeWidth(e){this.NodeWidth||(this.NodeWidth=new le),this.NodeWidth.value=e}getNodeHeight(){return this.NodeHeight??this.getType()==te.nodeType.comment?te.defaultCommentHeight:void 0}setNodeHeight(e){this.NodeHeight||(this.NodeHeight=new le),this.NodeHeight.value=e}getNodePosX(){return this.NodePosX?.value??0}setNodePosX(e){this.NodePosX||(this.NodePosX=new le),this.NodePosX.value=Math.round(e)}getNodePosY(){return this.NodePosY?.value??0}setNodePosY(e){this.NodePosY||(this.NodePosY=new le),this.NodePosY.value=Math.round(e)}isEvent(){return this.getClass()===te.nodeType.event||this.getClass()===te.nodeType.customEvent||!!this.getDelegatePin()}isDevelopmentOnly(){const e=this.getClass();return"DevelopmentOnly"===this.EnabledState?.toString()||e.includes("Debug",Math.max(0,e.lastIndexOf(".")))}hasHIDAttribute(){return this.InputKey??this.AxisKey??this.InputAxisKey}getDelegatePin(){return this.CustomProperties?.find((e=>"delegate"===e.PinType.PinCategory))}nodeDisplayName(){switch(this.getType()){case te.nodeType.componentBoundEvent:return`${ae.formatStringName(this.DelegatePropertyName)} (${this.ComponentPropertyName})`;case te.nodeType.dynamicCast:return this.TargetType?`Cast To ${this.TargetType.getName()}`:"Bad cast node";case te.nodeType.enumLiteral:return`Literal enum ${this.Enum.getName()}`;case te.nodeType.event:return`Event ${(this.EventReference?.MemberName??"").replace(/^Receive/,"")}`;case te.nodeType.executionSequence:return"Sequence";case te.nodeType.forEachElementInEnum:return`For Each ${this.Enum.getName()}`;case te.nodeType.forEachLoopWithBreak:return"For Each Loop with Break";case te.nodeType.ifThenElse:return"Branch";case te.nodeType.variableGet:return"";case te.nodeType.variableSet:return"SET"}const e=this.hasHIDAttribute();if(e){const t=e.toString();let i=He.keyName(t)??ae.formatStringName(t);return this.getClass()===te.nodeType.inputDebugKey?i="Debug Key "+i:this.getClass()===te.nodeType.getInputAxisKeyValue&&(i="Get "+i),i}if(this.getClass()===te.nodeType.macro)return ae.formatStringName(this.MacroGraphReference.getMacroName());let t=this.FunctionReference?.MemberName;if(t){const e=this.FunctionReference.MemberParent?.path??"";switch(t){case"AddKey":{let t=e.match(He.sequencerScriptingNameRegex);if(t)return`Add Key (${ae.formatStringName(t[1])})`}break;case"LineTraceSingle":return"Line Trace By Channel";case"LineTraceSingleByProfile":return"Line Trace By Profile"}switch(e){case"/Script/Engine.KismetMathLibrary":if(t.startsWith("Conv_"))return"";if(t.startsWith("Percent_"))return"%";if(t.startsWith("EqualEqual_"))return"==";const e=t.match(/[BF]([A-Z]\w+)/);switch(e&&(t=e[1]),t){case"Abs":return"ABS";case"Exp":return"e";case"LineTraceSingle":return"Line Trace By Channel";case"Max":case"MaxInt64":return"MAX";case"Min":case"MinInt64":return"MIN"}break;case"/Script/Engine.BlueprintSetLibrary":{const e=t.match(/Set_(\w+)/);if(e)return ae.formatStringName(e[1]).toUpperCase()}break;case"/Script/Engine.BlueprintMapLibrary":{const e=t.match(/Map_(\w+)/);if(e)return ae.formatStringName(e[1]).toUpperCase()}}return ae.formatStringName(t)}return ae.formatStringName(this.getNameAndCounter()[0])}nodeColor(){switch(this.getClass()){case te.nodeType.callFunction:return this.bIsPureFunc?te.nodeColors.green:te.nodeColors.blue;case te.nodeType.event:case te.nodeType.customEvent:case te.nodeType.inputKey:case te.nodeType.inputAxisKeyEvent:case te.nodeType.inputDebugKey:return te.nodeColors.red;case te.nodeType.enumLiteral:case te.nodeType.makeArray:case te.nodeType.makeMap:case te.nodeType.select:return te.nodeColors.green;case te.nodeType.executionSequence:case te.nodeType.ifThenElse:case te.nodeType.macro:return te.nodeColors.gray;case te.nodeType.dynamicCast:return te.nodeColors.turquoise}return this.bIsPureFunc?te.nodeColors.green:this.isEvent()?te.nodeColors.red:te.nodeColors.blue}nodeIcon(){switch(this.getType()){case te.nodeType.customEvent:return De.event;case te.nodeType.doN:return De.doN;case te.nodeType.dynamicCast:return De.cast;case te.nodeType.enumLiteral:return De.enum;case te.nodeType.event:return De.event;case te.nodeType.executionSequence:return De.sequence;case te.nodeType.forEachElementInEnum:return De.loop;case te.nodeType.forEachLoop:case te.nodeType.forEachLoopWithBreak:return De.forEachLoop;case te.nodeType.forLoop:case te.nodeType.forLoopWithBreak:return De.loop;case te.nodeType.ifThenElse:return De.branchNode;case te.nodeType.isValid:return De.questionMark;case te.nodeType.makeArray:return De.makeArray;case te.nodeType.makeMap:return De.makeMap;case te.nodeType.makeSet:return De.makeSet;case te.nodeType.select:return De.select;case te.nodeType.whileLoop:return De.loop}if(this.nodeDisplayName().startsWith("Break"))return De.breakStruct;if(this.getClass()===te.nodeType.macro)return De.macro;const e=this.hasHIDAttribute()?.toString();return e?e.includes("Mouse")?De.mouse:e.includes("Gamepad_Special")?De.keyboard:e.includes("Gamepad")||e.includes("Steam")?De.gamepad:e.includes("Touch")?De.touchpad:De.keyboard:this.getDelegatePin()?De.event:De.functionSymbol}}"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self&&self;function Oe(e){return e&&e.__esModule&&Object.prototype.hasOwnProperty.call(e,"default")?e.default:e}var Ve={exports:{}};"undefined"!=typeof self&&self;var ze=Oe(Ve.exports=function(e){var t={};function i(n){if(t[n])return t[n].exports;var s=t[n]={i:n,l:!1,exports:{}};return e[n].call(s.exports,s,s.exports,i),s.l=!0,s.exports}return i.m=e,i.c=t,i.d=function(e,t,n){i.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:n})},i.r=function(e){Object.defineProperty(e,"__esModule",{value:!0})},i.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return i.d(t,"a",t),t},i.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},i.p="",i(i.s=0)}([function(e,t,i){function n(e){if(!(this instanceof n))return new n(e);this._=e}var s=n.prototype;function r(e,t){for(var i=0;i>7),buf:function(e){var t=a((function(e,t,i,n){return e.concat(i===n.length-1?Buffer.from([t,0]).readUInt16BE(0):n.readUInt16BE(i))}),[],e);return Buffer.from(o((function(e){return(e<<1&65535)>>8}),t))}(i.buf)}})),i}function u(){return"undefined"!=typeof Buffer}function c(){if(!u())throw new Error("Buffer global does not exist; please use webpack if you need to parse Buffers in the browser.")}function h(e){c();var t=a((function(e,t){return e+t}),0,e);if(t%8!=0)throw new Error("The bits ["+e.join(", ")+"] add up to "+t+" which is not an even number of bytes; the total should be divisible by 8");var i,s=t/8,r=(i=function(e){return e>48},a((function(e,t){return e||(i(t)?t:e)}),null,e));if(r)throw new Error(r+" bit range requested exceeds 48 bit (6 byte) Number max.");return new n((function(t,i){var n=s+i;return n>t.length?C(i,s.toString()+" bytes"):E(n,a((function(e,t){var i=l(t,e.buf);return{coll:e.coll.concat(i.v),buf:i.buf}}),{coll:[],buf:t.slice(i,n)},e).coll)}))}function d(e,t){return new n((function(i,n){return c(),n+t>i.length?C(n,t+" bytes for "+e):E(n+t,i.slice(n,n+t))}))}function p(e,t){if("number"!=typeof(i=t)||Math.floor(i)!==i||t<0||t>6)throw new Error(e+" requires integer length in range [0, 6].");var i}function m(e){return p("uintBE",e),d("uintBE("+e+")",e).map((function(t){return t.readUIntBE(0,e)}))}function g(e){return p("uintLE",e),d("uintLE("+e+")",e).map((function(t){return t.readUIntLE(0,e)}))}function b(e){return p("intBE",e),d("intBE("+e+")",e).map((function(t){return t.readIntBE(0,e)}))}function v(e){return p("intLE",e),d("intLE("+e+")",e).map((function(t){return t.readIntLE(0,e)}))}function f(e){return e instanceof n}function y(e){return"[object Array]"==={}.toString.call(e)}function w(e){return u()&&Buffer.isBuffer(e)}function E(e,t){return{status:!0,index:e,value:t,furthest:-1,expected:[]}}function C(e,t){return y(t)||(t=[t]),{status:!1,index:-1,value:null,furthest:e,expected:t}}function S(e,t){if(!t)return e;if(e.furthest>t.furthest)return e;var i=e.furthest===t.furthest?function(e,t){if(function(){if(void 0!==n._supportsSet)return n._supportsSet;var e="undefined"!=typeof Set;return n._supportsSet=e,e}()&&Array.from){for(var i=new Set(e),s=0;s=0;){if(a in i){n=i[a].line,0===r&&(r=i[a].lineStart);break}("\n"===e.charAt(a)||"\r"===e.charAt(a)&&"\n"!==e.charAt(a+1))&&(s++,0===r&&(r=a+1)),a--}var o=n+s,l=t-r;return i[t]={line:o,lineStart:r},{offset:t,line:o+1,column:l+1}}function k(e){if(!f(e))throw new Error("not a parser: "+e)}function L(e,t){return"string"==typeof e?e.charAt(t):e[t]}function N(e){if("number"!=typeof e)throw new Error("not a number: "+e)}function T(e){if("function"!=typeof e)throw new Error("not a function: "+e)}function A(e){if("string"!=typeof e)throw new Error("not a string: "+e)}var M=2,B=3,D=8,$=5*D,H=4*D,O=" ";function V(e,t){return new Array(t+1).join(e)}function z(e,t,i){var n=t-e.length;return n<=0?e:V(i,n)+e}function I(e,t,i,n){return{from:e-t>0?e-t:0,to:e+i>n?n:e+i}}function R(e,t){var i,n,s,r,l,u=t.index,c=u.offset,h=1;if(c===e.length)return"Got the end of the input";if(w(e)){var d=c-c%D,p=c-d,m=I(d,$,H+D,e.length),g=o((function(e){return o((function(e){return z(e.toString(16),2,"0")}),e)}),function(e,t){var i=e.length,n=[],s=0;if(i<=t)return[e.slice()];for(var r=0;r=4&&(i+=1),h=2,s=o((function(e){return e.length<=4?e.join(" "):e.slice(0,4).join(" ")+" "+e.slice(4).join(" ")}),g),(l=(8*(r.to>0?r.to-1:r.to)).toString(16).length)<2&&(l=2)}else{var b=e.split(/\r\n|[\n\r\u2028\u2029]/);i=u.column-1,n=u.line-1,r=I(n,M,B,b.length),s=b.slice(r.from,r.to),l=r.to.toString().length}var v=n-r.from;return w(e)&&(l=(8*(r.to>0?r.to-1:r.to)).toString(16).length)<2&&(l=2),a((function(t,n,s){var a,o=s===v,u=o?"> ":O;return a=w(e)?z((8*(r.from+s)).toString(16),l,"0"):z((r.from+s+1).toString(),l," "),[].concat(t,[u+a+" | "+n],o?[O+V(" ",l)+" | "+z("",i," ")+V("^",h)]:[])}),[],s).join("\n")}function _(e,t){return["\n","-- PARSING FAILED "+V("-",50),"\n\n",R(e,t),"\n\n",(i=t.expected,1===i.length?"Expected:\n\n"+i[0]:"Expected one of the following: \n\n"+i.join(", ")),"\n"].join("");var i}function F(e){return void 0!==e.flags?e.flags:[e.global?"g":"",e.ignoreCase?"i":"",e.multiline?"m":"",e.unicode?"u":"",e.sticky?"y":""].join("")}function G(){for(var e=[].slice.call(arguments),t=e.length,i=0;i=2?N(t):t=0;var i=function(e){return RegExp("^(?:"+e.source+")",F(e))}(e),s=""+e;return n((function(e,n){var r=i.exec(e.slice(n));if(r){if(0<=t&&t<=r.length){var a=r[0],o=r[t];return E(n+a.length,o)}return C(n,"valid match group (0 to "+r.length+") in "+s)}return C(n,s)}))}function Y(e){return n((function(t,i){return E(i,e)}))}function q(e){return n((function(t,i){return C(i,e)}))}function J(e){if(f(e))return n((function(t,i){var n=e._(t,i);return n.index=i,n.value="",n}));if("string"==typeof e)return J(X(e));if(e instanceof RegExp)return J(Z(e));throw new Error("not a string, regexp, or parser: "+e)}function Q(e){return k(e),n((function(t,i){var n=e._(t,i),s=t.slice(i,n.index);return n.status?C(i,'not "'+s+'"'):E(i,null)}))}function ee(e){return T(e),n((function(t,i){var n=L(t,i);return i=e.length?C(t,"any character/byte"):E(t+1,L(e,t))})),re=n((function(e,t){return E(e.length,e.slice(t))})),ae=n((function(e,t){return t=0})).desc(t)},n.optWhitespace=he,n.Parser=n,n.range=function(e,t){return ee((function(i){return e<=i&&i<=t})).desc(e+"-"+t)},n.regex=Z,n.regexp=Z,n.sepBy=K,n.sepBy1=W,n.seq=G,n.seqMap=j,n.seqObj=function(){for(var e,t={},i=0,s=(e=arguments,Array.prototype.slice.call(e)),r=s.length,a=0;a255)throw new Error("Value specified to byte constructor ("+e+"=0x"+e.toString(16)+") is larger in value than a single byte.");var t=(e>15?"0x":"0x0")+e.toString(16);return n((function(i,n){var s=L(i,n);return s===e?E(n+1,s):C(n,t)}))},buffer:function(e){return d("buffer",e).map((function(e){return Buffer.from(e)}))},encodedString:function(e,t){return d("string",t).map((function(t){return t.toString(e)}))},uintBE:m,uint8BE:m(1),uint16BE:m(2),uint32BE:m(4),uintLE:g,uint8LE:g(1),uint16LE:g(2),uint32LE:g(4),intBE:b,int8BE:b(1),int16BE:b(2),int32BE:b(4),intLE:v,int8LE:v(1),int16LE:v(2),int32LE:v(4),floatBE:d("floatBE",4).map((function(e){return e.readFloatBE(0)})),floatLE:d("floatLE",4).map((function(e){return e.readFloatLE(0)})),doubleBE:d("doubleBE",8).map((function(e){return e.readDoubleBE(0)})),doubleLE:d("doubleLE",8).map((function(e){return e.readDoubleLE(0)}))},e.exports=n}]));class Ie extends oe{static attributes={lookbehind:{value:"",showDefault:!1,ignore:!0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.lookbehind}}let Re=ze;class _e{static Regex={ByteInteger:/0*(?:25[0-5]|2[0-4]\d|1?\d?\d|)(?!\d)/,HexDigit:/[0-9a-fA-F]/,InlineOptWhitespace:/[^\S\n]*/,InlineWhitespace:/[^\S\n]+/,InsideString:/(?:[^"\\]|\\.)*/,Integer:/[\-\+]?\d+/,MultilineWhitespace:/\s*\n\s*/,Number:/[-\+]?\d+(?:\.\d+)?/,RealUnit:/^\+?(?:0(?:\.\d+)?|1(?:\.0+)?)(?![\.\d])/,Symbol:/[a-zA-Z_]\w*/,Word:/[a-zA-Z_]+/};static regexMap(e,t){const i=RegExp("^(?:"+e.source+")",e.flags),n=""+e;return Re(((e,s)=>{const r=i.exec(e.slice(s));return r?Re.makeSuccess(s+r[0].length,t(r)):Re.makeFailure(s,n)}))}static getStringParser(e){return Re(((t,i)=>{var n=i+e.length,s=t.slice(i,n);return s===e?makeSuccess(n,s):makeFailure(i,expected)}))}static getGrammarForType(e,t,i=e.AttributeAnyValue){if(t.constructor===Object){let n,s=t.type;return n=s instanceof Array?_e.getGrammarForType(e,s[0]).trim(Re.optWhitespace).sepBy(Re.string(",")).skip(Re.regex(/,?\s*/)).wrap(Re.string("("),Re.string(")")):s instanceof re?s.types.map((t=>_e.getGrammarForType(e,ae.getType(t)))).reduce(((t,i)=>i&&t!==e.AttributeAnyValue?t.or(i):e.AttributeAnyValue)):_e.getGrammarForType(e,s,i),!t.serialized||s instanceof String||(n=n.wrap(Re.string('"'),Re.string('"'))),t.nullable&&(n=n.or(e.Null)),n}switch(t){case BigInt:return e.BigInt;case Boolean:return e.Boolean;case ue:return e.Byte;case he:return e.Enum;case me:return e.FunctionReference;case de:return e.Guid;case ge:return e.Identifier;case be:return e.Integer64;case le:return e.Integer;case ve:return e.InvariantText;case fe:return e.KeyBinding;case we:return e.LinearColor;case Ee:return e.LocalizedText;case Ce:return e.MacroGraphReference;case Number:return e.Number;case pe:return e.ObjectReference;case Se:return e.PathSymbol;case Be:return e.Pin;case Pe:return e.PinReference;case xe:return e.PinType;case ye:return e.RealUnit;case ke:return e.Rotator;case Le:return e.SimpleSerializationRotator;case Te:return e.SimpleSerializationVector2D;case Me:return e.SimpleSerializationVector;case String:return e.String;case ce:return e.Symbol;case $e:return e.VariableReference;case Ne:return e.Vector2D;case Ae:return e.Vector;default:return i}}static ReferencePath=(e,t)=>Re.alt(t,Re.seq(Re.string("/"),t.map((e=>e.toString())).sepBy1(Re.string(".")).tieWith(".").sepBy1(Re.string(":")).tieWith(":")).tie().atLeast(2).tie());static createAttributeGrammar=(e,t,i=Re.string("=").trim(Re.optWhitespace))=>e.AttributeName.skip(i).chain((i=>{const n=i.split("."),s=ae.objectGet(t.attributes,n);return(s?s.constructor===Object&&s.serialized?e.String:_e.getGrammarForType(e,s,e.AttributeAnyValue):e.AttributeAnyValue).map((e=>t=>ae.objectSet(t,n,e,!0)))}));static createEntityGrammar=(e,t,i=!0)=>Re.seqMap(t.lookbehind?Re.seq(Re.string(t.lookbehind),Re.optWhitespace,Re.string("(")):Re.string("("),_e.createAttributeGrammar(e,t).trim(Re.optWhitespace).sepBy(Re.string(",")).skip(Re.regex(/,?/).then(Re.optWhitespace)),Re.string(")"),((e,t,i)=>{let n={};return t.forEach((e=>e(n))),n})).chain((e=>{let n=Object.keys(e);if(Object.keys(t.attributes).filter((e=>t.attributes[e].expected)).find((e=>!n.includes(e))))return Re.fail();const s=Object.keys(e).filter((e=>!(e in t.attributes))).length;return!i&&s>0||i&&s+.5>Math.sqrt(n)?Re.fail():Re.succeed().map((()=>new t(e)))}));Null=e=>Re.regex(new RegExp(String.raw`\(${_e.Regex.InlineOptWhitespace.source}\)`)).map((()=>null)).desc("null: ()");Boolean=e=>_e.regexMap(/(true)|false/i,(e=>!!e[1])).desc("either True or False");Number=e=>Re.regex(_e.Regex.Number).map(Number).desc("a number");BigInt=e=>Re.regex(_e.Regex.Integer).map((e=>BigInt(e))).desc("a big integer");RealUnit=e=>Re.regex(_e.Regex.RealUnit).map(Number).desc("a number between 0 and 1");NaturalNumber=e=>_e.regexMap(/\d+/,(e=>parseInt(e[0]))).desc("a natural number");ColorNumber=e=>Re.regexp(_e.Regex.ByteInteger).desc("a number between 0 and 255");Word=e=>Re.regexp(_e.Regex.Word).desc("a word");String=e=>Re.regexp(new RegExp(`"(${_e.Regex.InsideString.source})"`),1).map(ae.unescapeString).desc('string (with possibility to escape the quote using ")');AttributeName=e=>Re.regexp(new RegExp(String.raw`(?:(?:^|(?Re.string("None").map((()=>new pe({type:"None",path:""}))).desc("none");Integer64=e=>e.BigInt.map((e=>new be(e))).desc("an integer64");Integer=e=>Re.regex(_e.Regex.Integer).map((e=>new le(e))).desc("an integer");Byte=e=>Re.regex(_e.Regex.ByteInteger).map((e=>new ue(parseInt(e)))).desc("a Byte");Guid=e=>Re.regexp(new RegExp(`${_e.Regex.HexDigit.source}{32}`)).map((e=>new de({value:e}))).desc("32 digit hexadecimal value");Identifier=e=>Re.regex(/\w+/).map((e=>new ge(e)));PathSymbol=e=>Re.regex(/\w+/).map((e=>new Se({value:e})));PathSymbolOptSpaces=e=>Re.regex(/(?:(?:^|(?new Se({value:e})));Symbol=e=>Re.regex(_e.Regex.Symbol).map((e=>new ce({value:e})));Enum=e=>Re.regex(_e.Regex.Symbol).map((e=>new he({value:e})));ObjectReference=e=>Re.alt(e.None,...[_e.ReferencePath(e,e.PathSymbolOptSpaces).map((e=>new pe({type:"",path:e})))].flatMap((e=>[e.wrap(Re.string('"'),Re.string('"')),e.wrap(Re.string("'\""),Re.string("\"'"))])),Re.seqMap(_e.ReferencePath(e,e.PathSymbolOptSpaces),Re.optWhitespace,Re.alt(...[_e.ReferencePath(e,e.PathSymbolOptSpaces)].flatMap((e=>[e.wrap(Re.string('"'),Re.string('"')),e.wrap(Re.string("'\""),Re.string("\"'"))]))),((e,t,i)=>new pe({type:e,path:i}))),_e.ReferencePath(e,e.PathSymbol).map((e=>new pe({type:"",path:e}))),e.Word.map((e=>new pe({type:e,path:""}))));LocalizedText=e=>_e.regexMap(new RegExp(String.raw`${Ee.lookbehind}\s*\(`+String.raw`\s*"(${_e.Regex.InsideString.source})"\s*,`+String.raw`\s*"(${_e.Regex.InsideString.source})"\s*,`+String.raw`\s*"(${_e.Regex.InsideString.source})"\s*\)`),(e=>new Ee({namespace:e[1],key:e[2],value:e[3]})));InvariantText=e=>e.String.trim(Re.optWhitespace).wrap(Re.string(ve.lookbehind).skip(Re.optWhitespace).skip(Re.string("(")),Re.string(")")).map((e=>new ve({value:e})));AttributeAnyValue=e=>Re.alt(e.Boolean,e.Guid,e.None,e.Null,e.Number,e.String,e.LocalizedText,e.InvariantText,e.PinReference,e.Vector,e.LinearColor,e.Vector2D,e.UnknownKeys,e.ObjectReference,e.Symbol);PinReference=e=>Re.seqMap(e.PathSymbol,Re.whitespace,e.Guid,((e,t,i)=>new Pe({objectName:e,pinGuid:i})));PinType=e=>_e.createEntityGrammar(e,xe,!0);Vector2D=e=>_e.createEntityGrammar(e,Ne,!1);Vector=e=>_e.createEntityGrammar(e,Ae,!1);Rotator=e=>_e.createEntityGrammar(e,ke,!1);SimpleSerializationRotator=e=>Re.seqMap(e.Number,Re.string(",").trim(Re.optWhitespace),e.Number,Re.string(",").trim(Re.optWhitespace),e.Number,((e,t,i,n,s)=>new Le({R:s,P:e,Y:i})));SimpleSerializationVector2D=e=>Re.seqMap(e.Number,Re.string(",").trim(Re.optWhitespace),e.Number,((e,t,i)=>new Te({X:e,Y:i})));SimpleSerializationVector=e=>Re.seqMap(e.Number,Re.string(",").trim(Re.optWhitespace),e.Number,Re.string(",").trim(Re.optWhitespace),e.Number,((e,t,i,n,s)=>new Me({X:e,Y:i,Z:s})));LinearColor=e=>_e.createEntityGrammar(e,we,!1);FunctionReference=e=>_e.createEntityGrammar(e,me);VariableReference=e=>_e.createEntityGrammar(e,$e);MacroGraphReference=e=>_e.createEntityGrammar(e,Ce);KeyBinding=e=>Re.alt(e.Identifier.map((e=>new fe({Key:e}))),_e.createEntityGrammar(e,fe));Pin=e=>_e.createEntityGrammar(e,Be);CustomProperties=e=>Re.regex(/CustomProperties\s+/).then(e.Pin).map((e=>t=>{let i=ae.objectGet(t,["CustomProperties"],[]);i.push(e),ae.objectSet(t,["CustomProperties"],i,!0)}));Object=e=>Re.seqMap(Re.regexp(/Begin\s+Object\s+/),Re.alt(e.CustomProperties,_e.createAttributeGrammar(e,He)).sepBy1(Re.whitespace),Re.regexp(/\s+End\s+Object/),((e,t,i)=>{let n={};return t.forEach((e=>e(n))),new He(n)}));MultipleObject=e=>e.Object.sepBy1(Re.whitespace).trim(Re.optWhitespace);LinearColorFromHex=e=>Re.string("#").then(e.HexDigit.times(2).tie().times(3,4)).trim(Re.optWhitespace).map((([e,t,i,n])=>new we({R:parseInt(e,16)/255,G:parseInt(t,16)/255,B:parseInt(i,16)/255,A:n?parseInt(n,16)/255:1})));LinearColorFromRGBList=e=>Re.seqMap(e.ColorNumber,Re.string(",").skip(Re.optWhitespace),e.ColorNumber,Re.string(",").skip(Re.optWhitespace),e.ColorNumber.map(Number),((e,t,i,n,s)=>new we({R:e/255,G:i/255,B:s/255,A:1})));LinearColorFromRGB=e=>Re.string("rgb").then(e.LinearColorFromRGBList.wrap(Re.regex(/\(\s*/),Re.regex(/\s*\)/)));LinearColorFromRGBA=e=>Re.string("rgba").then(Re.seqMap(e.ColorNumber,Re.string(",").skip(Re.optWhitespace),e.ColorNumber,Re.string(",").skip(Re.optWhitespace),e.ColorNumber.map(Number),Re.string(",").skip(Re.optWhitespace),Re.regex(/0?\.\d+|[01]/).map(Number),((e,t,i,n,s,r,a)=>new we({R:e/255,G:i/255,B:s/255,A:a}))).wrap(Re.regex(/\(\s*/),Re.regex(/\s*\)/)));LinearColorFromAnyColor=e=>Re.alt(e.LinearColorFromRGBList,e.LinearColorFromHex,e.LinearColorFromRGB,e.LinearColorFromRGBA);UnknownKeys=e=>Re.seqMap(Re.regex(/\w*\s*/).skip(Re.string("(")),Re.seqMap(e.AttributeName,Re.string("=").trim(Re.optWhitespace),e.AttributeAnyValue,((e,t,i)=>t=>ae.objectSet(t,e.split("."),i,!0))).trim(Re.optWhitespace).sepBy(Re.string(",")).skip(Re.regex(/,?/).then(Re.optWhitespace)),Re.string(")"),((e,t,i)=>{let n={};t.forEach((e=>e(n)));let s=new Ie(n);return e&&(s.lookbehind=e),s}))}class Fe{static grammar=ze.createLanguage(new _e);constructor(e,t="",i=",",n=!1,s="=",r=(e=>e.join("."))){this.entityType=e,this.attributePrefix=t,this.attributeSeparator=i,this.trailingSeparator=n,this.attributeValueConjunctionSign=s,this.attributeKeyPrinter=r}deserialize(e){return this.read(e)}serialize(e,t=!1,i=e){return this.write(i,e,t)}read(e){throw new Error("Not implemented")}write(e,t,i){throw new Error("Not implemented")}writeValue(e,t,i,n){const s=ae.getType(t),r=ne.getSerializer(s);if(!r)throw new Error(`Unknown value type "${s.name}", a serializer must be registered in the SerializerFactory class, check initializeSerializerFactory.js`);return r.write(t instanceof oe?t:e,t,n)}subWrite(e,t,i,n){let s="",r=t.concat("");const a=r.length-1,o=i.constructor.attributes,l=o?ae.mergeArrays(Object.keys(o),Object.keys(i)):Object.keys(i);for(const t of l){r[a]=t;const o=i[t];if(o?.constructor===Object)s+=(s.length?this.attributeSeparator:"")+this.subWrite(e,r,o,n);else if(void 0!==o&&this.showProperty(e,i,r,o)){const t=ae.isSerialized(e,r);s+=(s.length?this.attributeSeparator:"")+this.attributePrefix+this.attributeKeyPrinter(r)+this.attributeValueConjunctionSign+(t?`"${this.writeValue(e,o,r,!0)}"`:this.writeValue(e,o,r,n))}}return this.trailingSeparator&&s.length&&1===r.length&&(s+=this.attributeSeparator),s}showProperty(e,t,i,n){const s=this.entityType.attributes,r=ae.objectGet(s,i);return r?.constructor!==Object||!r.ignored&&(!ae.equals(r.value,n)||r.showDefault)}}class Ge extends Fe{constructor(){super(He," ","\n",!1)}showProperty(e,t,i,n){switch(i.toString()){case"Class":case"Name":case"CustomProperties":return!1}return super.showProperty(e,t,i,n)}read(e){const t=Fe.grammar.Object.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}readMultiple(e){const t=Fe.grammar.MultipleObject.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}write(e,t,i){return`Begin Object Class=${t.Class.path} Name=${this.writeValue(e,t.Name,["Name"],i)}\n${this.subWrite(e,[],t,i)+t.CustomProperties.map((e=>this.attributeSeparator+this.attributePrefix+"CustomProperties "+ne.getSerializer(Be).serialize(e))).join("")}\nEnd Object\n`}}class je extends ie{static#l=new Ge;#u;constructor(e,t,i={}){i.listenOnFocus??=!0,i.unlistenOnTextEdit??=!0,super(e,t,i);let n=this;this.#u=e=>n.copied()}listenEvents(){window.addEventListener("copy",this.#u)}unlistenEvents(){window.removeEventListener("copy",this.#u)}getSerializedText(){return this.blueprint.getNodes(!0).map((e=>je.#l.serialize(e.entity,!1))).join("")}copied(){const e=this.getSerializedText();navigator.clipboard.writeText(e)}}class Ue{element;get blueprint(){return this.element.blueprint}#c=[];get inputObjects(){return this.#c}initialize(e){this.element=e}createInputObjects(){return[]}getInputObject(e){return this.inputObjects.find((t=>t.constructor==e))}setup(){this.#c.forEach((e=>e.setup()))}cleanup(){this.#c.forEach((e=>e.cleanup()))}willUpdate(e){}update(e){}render(){return $``}firstUpdated(e){}updated(e){}inputSetup(){this.#c=this.createInputObjects()}}class Ke extends ie{#h;constructor(e,t,i={}){i.activateAnyKey??=!1,i.activationKeys??=[],i.consumeEvent??=!0,i.listenOnFocus??=!0,i.unlistenOnTextEdit??=!0,i.activationKeys instanceof Array||(i.activationKeys=[i.activationKeys]),i.activationKeys=i.activationKeys.map((e=>{if(e instanceof fe)return e;if(e.constructor===String){const t=Fe.grammar.KeyBinding.parse(e);if(t.status)return t.value}throw new Error("Unexpected key value")})),super(e,t,i),this.#h=this.options.activationKeys??[];let n=this;this.keyDownHandler=e=>{(this.options.activateAnyKey||n.#h.some((t=>(e=>e.bShift||"LeftShift"==e.Key||"RightShift"==e.Key)(t)==e.shiftKey&&(e=>e.bCtrl||"LeftControl"==e.Key||"RightControl"==e.Key)(t)==e.ctrlKey&&(e=>e.bAlt||"LeftAlt"==e.Key||"RightAlt"==e.Key)(t)==e.altKey&&te.Keys[t.Key]==e.code)))&&(i.consumeEvent&&(e.preventDefault(),e.stopImmediatePropagation()),n.fire(),document.removeEventListener("keydown",n.keyDownHandler),document.addEventListener("keyup",n.keyUpHandler))},this.keyUpHandler=e=>{(this.options.activateAnyKey||n.#h.some((t=>t.bShift&&"Shift"==e.key||t.bCtrl&&"Control"==e.key||t.bAlt&&"Alt"==e.key||t.bCmd&&"Meta"==e.key||te.Keys[t.Key]==e.code)))&&(i.consumeEvent&&e.stopImmediatePropagation(),n.unfire(),document.removeEventListener("keyup",this.keyUpHandler),document.addEventListener("keydown",this.keyDownHandler))}}listenEvents(){document.addEventListener("keydown",this.keyDownHandler)}unlistenEvents(){document.removeEventListener("keydown",this.keyDownHandler)}fire(){}unfire(){}}class We extends Ke{constructor(e,t,i={}){i.activationKeys=te.deleteNodesKeyboardKey,super(e,t,i)}fire(){this.blueprint.removeGraphElement(...this.blueprint.getNodes(!0))}}class Xe extends ie{constructor(e,t,i={}){i.ignoreTranslateCompensate??=!1,i.ignoreScale??=!1,i.movementSpace??=t.getGridDOMElement()??document.documentElement,super(e,t,i),this.movementSpace=i.movementSpace}locationFromEvent(e){const t=ae.convertLocation([e.clientX,e.clientY],this.movementSpace,this.options.ignoreScale);return this.options.ignoreTranslateCompensate?t:this.blueprint.compensateTranslation(t[0],t[1])}}class Ze extends Xe{#d=e=>{e.preventDefault();const t=this.locationFromEvent(e);this.wheel(Math.sign(e.deltaY*te.mouseWheelFactor),t)};#p=e=>e.preventDefault();constructor(e,t,i={}){i.listenOnFocus=!0,i.strictTarget??=!1,super(e,t,i),this.strictTarget=i.strictTarget}listenEvents(){this.movementSpace.addEventListener("wheel",this.#d,!1),this.movementSpace.parentElement?.addEventListener("wheel",this.#p)}unlistenEvents(){this.movementSpace.removeEventListener("wheel",this.#d,!1),this.movementSpace.parentElement?.removeEventListener("wheel",this.#p)}wheel(e,t){}}class Ye extends Ze{#m=!1;get enableZoonIn(){return this.#m}set enableZoonIn(e){e!=this.#m&&(this.#m=e)}wheel(e,t){let i=this.blueprint.getZoom();e=-e,!this.enableZoonIn&&0==i&&e>0||(i+=e,this.blueprint.setZoom(i,t))}}class qe extends Ke{#g;constructor(e,t,i={}){i.activationKeys=te.enableZoomIn,super(e,t,i)}fire(){this.#g=this.blueprint.getInputObject(Ye),this.#g.enableZoonIn=!0}unfire(){this.#g.enableZoonIn=!1}}class Je extends Ke{constructor(e,t,i={}){i.activationKeys=te.selectAllKeyboardKey,super(e,t,i)}fire(){this.blueprint.selectAll()}}class Qe extends Q{#b=[];#t;get blueprint(){return this.#t}set blueprint(e){this.#t=e}#v;get entity(){return this.#v}set entity(e){this.#v=e}#f;get template(){return this.#f}isInitialized=!1;isSetup=!1;inputObjects=[];initialize(e,t){this.requestUpdate(),this.#v=e,this.#f=t,this.#f.initialize(this),this.isConnected&&this.updateComplete.then((()=>this.setup())),this.isInitialized=!0}connectedCallback(){super.connectedCallback(),this.blueprint=this.closest("ueb-blueprint"),this.isInitialized&&(this.requestUpdate(),this.updateComplete.then((()=>this.setup())))}disconnectedCallback(){super.disconnectedCallback(),this.isSetup&&this.updateComplete.then((()=>this.cleanup())),this.acknowledgeDelete()}createRenderRoot(){return this}shouldUpdate(e){return this.isInitialized&&this.isConnected}setup(){this.template.setup(),this.isSetup=!0}cleanup(){this.template.cleanup(),this.isSetup=!1}willUpdate(e){super.willUpdate(e),this.template.willUpdate(e)}update(e){super.update(e),this.template.update(e)}render(){return this.template.render()}firstUpdated(e){super.firstUpdated(e),this.template.firstUpdated(e),this.template.inputSetup()}updated(e){super.updated(e),this.template.updated(e);for(const t of this.#b)t(e);this.#b=[]}addNextUpdatedCallbacks(e,t=!1){this.#b.push(e),t&&this.requestUpdate()}acknowledgeDelete(){let e=new CustomEvent(te.removeEventName);this.dispatchEvent(e)}isSameGraph(e){return this.blueprint&&this.blueprint==e?.blueprint}getInputObject(e){return this.template.inputObjects.find((t=>t.constructor==e))}}class et extends Qe{static properties={...super.properties,locationX:{type:Number,attribute:!1},locationY:{type:Number,attribute:!1},sizeX:{type:Number,attribute:!1},sizeY:{type:Number,attribute:!1}};static dragEventName=te.dragEventName;static dragGeneralEventName=te.dragGeneralEventName;constructor(){super(),this.locationX=0,this.locationY=0,this.sizeX=0,this.sizeY=0}computeSizes(){const e=this.getBoundingClientRect();this.sizeX=this.blueprint.scaleCorrect(e.width),this.sizeY=this.blueprint.scaleCorrect(e.height)}firstUpdated(e){super.firstUpdated(e),this.computeSizes()}setLocation(e,t,i=!0){const n=e-this.locationX,s=t-this.locationY;if(this.locationX=e,this.locationY=t,this.blueprint&&i){const e=new CustomEvent(this.constructor.dragEventName,{detail:{value:[n,s]},bubbles:!1,cancelable:!0});this.dispatchEvent(e)}}addLocation(e,t,i=!0){this.setLocation(this.locationX+e,this.locationY+t,i)}acknowledgeDrag(e){const t=new CustomEvent(this.constructor.dragGeneralEventName,{detail:{value:e},bubbles:!0,cancelable:!0});this.dispatchEvent(t)}snapToGrid(){const e=ae.snapToGrid(this.locationX,this.locationY,te.gridSize);this.locationX==e[0]&&this.locationY==e[1]||this.setLocation(e[0],e[1])}topBoundary(e=!1){return this.template.topBoundary(e)}rightBoundary(e=!1){return this.template.rightBoundary(e)}bottomBoundary(e=!1){return this.template.bottomBoundary(e)}leftBoundary(e=!1){return this.template.leftBoundary(e)}}class tt extends Xe{#y=e=>{if(this.blueprint.setFocused(!0),e.button===this.options.clickButton)this.options.strictTarget&&e.target!=e.currentTarget||(this.options.consumeEvent&&e.stopImmediatePropagation(),this.#w.addEventListener("mousemove",this.#E),document.addEventListener("mouseup",this.#C),this.clickedPosition=this.locationFromEvent(e),this.blueprint.mousePosition[0]=this.clickedPosition[0],this.blueprint.mousePosition[1]=this.clickedPosition[1],this.target instanceof et&&(this.clickedOffset=[this.clickedPosition[0]-this.target.locationX,this.clickedPosition[1]-this.target.locationY]),this.clicked(this.clickedPosition));else this.options.exitAnyButton||this.#C(e)};#E=e=>{this.options.consumeEvent&&e.stopImmediatePropagation(),this.#w.removeEventListener("mousemove",this.#E),this.#w.addEventListener("mousemove",this.#S);const t=this.getEvent(te.trackingMouseEventName.begin);this.#P=0==this.target.dispatchEvent(t);const i=this.locationFromEvent(e);this.lastLocation=ae.snapToGrid(this.clickedPosition[0],this.clickedPosition[1],this.stepSize),this.startDrag(i),this.started=!0};#S=e=>{this.options.consumeEvent&&e.stopImmediatePropagation();const t=this.locationFromEvent(e),i=[e.movementX,e.movementY];if(this.dragTo(t,i),this.#P&&(this.blueprint.mousePosition=t),this.options.scrollGraphEdge){const e=Math.sqrt(i[0]*i[0]+i[1]*i[1]),n=this.blueprint.scaleCorrect(te.edgeScrollThreshold),s=this.blueprint.template.gridLeftVisibilityBoundary()+n,r=this.blueprint.template.gridRightVisibilityBoundary()-n;let a=0;t[0]r&&(a=t[0]-r);const o=this.blueprint.template.gridTopVisibilityBoundary()+n,l=this.blueprint.template.gridBottomVisibilityBoundary()-n;let u=0;t[1]l&&(u=t[1]-l),a=ae.clamp(this.blueprint.scaleCorrectReverse(a)**3*e*.6,-20,20),u=ae.clamp(this.blueprint.scaleCorrectReverse(u)**3*e*.6,-20,20),this.blueprint.scrollDelta(a,u)}};#C=e=>{if(!this.options.exitAnyButton||e.button==this.options.clickButton){if(this.options.consumeEvent&&e.stopImmediatePropagation(),this.#w.removeEventListener("mousemove",this.#E),this.#w.removeEventListener("mousemove",this.#S),document.removeEventListener("mouseup",this.#C),this.started&&this.endDrag(),this.unclicked(),this.#P){const e=this.getEvent(te.trackingMouseEventName.end);this.target.dispatchEvent(e),this.#P=!1}this.started=!1}};#P=!1;#w;#x;clickedOffset=[0,0];clickedPosition=[0,0];lastLocation=[0,0];started=!1;stepSize=1;constructor(e,t,i={}){i.clickButton??=0,i.consumeEvent??=!0,i.draggableElement??=e,i.exitAnyButton??=!0,i.moveEverywhere??=!1,i.movementSpace??=t?.getGridDOMElement(),i.repositionOnClick??=!1,i.scrollGraphEdge??=!1,i.strictTarget??=!1,super(e,t,i),this.stepSize=parseInt(i?.stepSize??te.gridSize),this.#w=this.options.moveEverywhere?document.documentElement:this.movementSpace,this.#x=this.options.draggableElement,this.listenEvents()}listenEvents(){super.listenEvents(),this.#x.addEventListener("mousedown",this.#y),2==this.options.clickButton&&this.#x.addEventListener("contextmenu",(e=>e.preventDefault()))}unlistenEvents(){super.unlistenEvents(),this.#x.removeEventListener("mousedown",this.#y)}getEvent(e){return new CustomEvent(e,{detail:{tracker:this},bubbles:!0,cancelable:!0})}clicked(e){}startDrag(e){}dragTo(e,t){}endDrag(){}unclicked(e){}}class it extends tt{startDrag(){this.blueprint.scrolling=!0}dragTo(e,t){this.blueprint.scrollDelta(-t[0],-t[1])}endDrag(){this.blueprint.scrolling=!1}}class nt extends Xe{#k=null;#L=e=>{e.preventDefault(),this.blueprint.mousePosition=this.locationFromEvent(e)};#N=e=>{this.#k||(e.preventDefault(),this.#k=e.detail.tracker,this.unlistenMouseMove())};#T=e=>{this.#k==e.detail.tracker&&(e.preventDefault(),this.#k=null,this.listenMouseMove())};constructor(e,t,i={}){i.listenOnFocus=!0,super(e,t,i)}listenMouseMove(){this.target.addEventListener("mousemove",this.#L)}unlistenMouseMove(){this.target.removeEventListener("mousemove",this.#L)}listenEvents(){this.listenMouseMove(),this.blueprint.addEventListener(te.trackingMouseEventName.begin,this.#N),this.blueprint.addEventListener(te.trackingMouseEventName.end,this.#T)}unlistenEvents(){this.unlistenMouseMove(),this.blueprint.removeEventListener(te.trackingMouseEventName.begin,this.#N),this.blueprint.removeEventListener(te.trackingMouseEventName.end,this.#T)}}class st{static#A=new Map;static registerElement(e,t){st.#A.set(e,t)}static getConstructor(e){return st.#A.get(e)}}class rt extends ie{static#l=new Ge;#M;constructor(e,t,i={}){i.listenOnFocus??=!0,i.unlistenOnTextEdit??=!0,super(e,t,i);let n=this;this.#M=e=>n.pasted(e.clipboardData.getData("Text"))}listenEvents(){window.addEventListener("paste",this.#M)}unlistenEvents(){window.removeEventListener("paste",this.#M)}pasted(e){let t=0,i=0,n=0,s=rt.#l.readMultiple(e).map((e=>{let s=st.getConstructor("ueb-node").newObject(e);return t+=s.locationY,i+=s.locationX,++n,s}));t/=n,i/=n,s.length>0&&this.blueprint.unselectAll();let r=this.blueprint.mousePosition;return s.forEach((e=>{e.addLocation(r[0]-i,r[1]-t),e.snapToGrid(),e.setSelected(!0)})),this.blueprint.addGraphElement(...s),!0}}class at extends tt{constructor(e,t,i={}){i.scrollGraphEdge??=!0,super(e,t,i),this.selectorElement=this.blueprint.template.selectorElement}startDrag(){this.selectorElement.beginSelect(this.clickedPosition)}dragTo(e,t){this.selectorElement.selectTo(e)}endDrag(){this.started&&this.selectorElement.endSelect()}unclicked(){this.started||this.blueprint.unselectAll()}}class ot extends ie{#B=e=>this.clickedSomewhere(e.target);constructor(e,t,i={}){i.listenOnFocus=!0,super(e,t,i),this.blueprint.focus&&document.addEventListener("click",this.#B)}clickedSomewhere(e){e.closest("ueb-blueprint")||this.blueprint.setFocused(!1)}listenEvents(){document.addEventListener("click",this.#B)}unlistenEvents(){document.removeEventListener("click",this.#B)}}class lt extends Ue{static styleVariables={"--ueb-font-size":`${te.fontSize}`,"--ueb-grid-axis-line-color":`${te.gridAxisLineColor}`,"--ueb-grid-expand":`${te.expandGridSize}px`,"--ueb-grid-line-color":`${te.gridLineColor}`,"--ueb-grid-line-width":`${te.gridLineWidth}px`,"--ueb-grid-set-line-color":`${te.gridSetLineColor}`,"--ueb-grid-set":`${te.gridSet}`,"--ueb-grid-size":`${te.gridSize}px`,"--ueb-link-min-width":`${te.linkMinWidth}`,"--ueb-node-radius":`${te.nodeRadius}px`};#D=new ResizeObserver((e=>{const t=e.find((e=>e.target===this.viewportElement))?.devicePixelContentBoxSize?.[0];t&&(this.viewportSize[0]=t.inlineSize,this.viewportSize[1]=t.blockSize)}));headerElement;overlayElement;viewportElement;selectorElement;gridElement;linksContainerElement;nodesContainerElement;viewportSize=[0,0];initialize(e){super.initialize(e),this.element.style.cssText=Object.entries(lt.styleVariables).map((([e,t])=>`${e}:${t};`)).join("")}setup(){super.setup(),this.#D.observe(this.viewportElement,{box:"device-pixel-content-box"});const e=this.viewportElement.getBoundingClientRect();this.viewportSize[0]=e.width,this.viewportSize[1]=e.height,this.blueprint.nodes.length>0&&(this.blueprint.requestUpdate(),this.blueprint.updateComplete.then((()=>this.centerContentInViewport())))}cleanup(){super.cleanup(),this.#D.unobserve(this.viewportElement)}createInputObjects(){return[...super.createInputObjects(),new je(this.element.getGridDOMElement(),this.element),new rt(this.element.getGridDOMElement(),this.element),new We(this.element.getGridDOMElement(),this.element),new Je(this.element.getGridDOMElement(),this.element),new Ye(this.element.getGridDOMElement(),this.element),new at(this.element.getGridDOMElement(),this.element,{clickButton:0,exitAnyButton:!0,moveEverywhere:!0}),new it(this.element.getGridDOMElement(),this.element,{clickButton:2,exitAnyButton:!1,moveEverywhere:!0}),new ot(this.element.getGridDOMElement(),this.element),new nt(this.element.getGridDOMElement(),this.element),new qe(this.element.getGridDOMElement(),this.element)]}render(){return $`
Zoom ${0==this.element.zoom?"1:1":(this.element.zoom>0?"+":"")+this.element.zoom}
`}firstUpdated(e){super.firstUpdated(e),this.headerElement=this.element.querySelector(".ueb-viewport-header"),this.overlayElement=this.element.querySelector(".ueb-viewport-overlay"),this.viewportElement=this.element.querySelector(".ueb-viewport-body"),this.selectorElement=this.element.querySelector("ueb-selector"),this.gridElement=this.viewportElement.querySelector(".ueb-grid"),this.linksContainerElement=this.element.querySelector("[data-links]"),this.linksContainerElement.append(...this.element.getLinks()),this.nodesContainerElement=this.element.querySelector("[data-nodes]"),this.nodesContainerElement.append(...this.element.getNodes()),this.viewportElement.scroll(te.expandGridSize,te.expandGridSize)}willUpdate(e){super.willUpdate(e),this.headerElement&&e.has("zoom")&&(this.headerElement.classList.add("ueb-zoom-changed"),this.headerElement.addEventListener("animationend",(()=>this.headerElement.classList.remove("ueb-zoom-changed"))))}updated(e){if(super.updated(e),(e.has("scrollX")||e.has("scrollY"))&&this.viewportElement.scroll(this.element.scrollX,this.element.scrollY),e.has("zoom")){this.element.style.setProperty("--ueb-scale",this.blueprint.getScale());const t=e.get("zoom"),i=Math.min(t,this.element.zoom),n=Math.max(t,this.element.zoom),s=ae.range(i,n),r=e=>`ueb-zoom-${e}`;te<0)).map(r)),this.element.classList.add(...s.filter((e=>e>0)).map(r))):(this.element.classList.remove(...s.filter((e=>e>0)).map(r)),this.element.classList.add(...s.filter((e=>e<0)).map(r)))}}getCommentNodes(e=!1){return this.element.querySelectorAll(`ueb-node[data-type="${te.nodeType.comment}"]${e?'[data-selected="true"]':""}`)}getPin(e){return this.element.querySelector(`ueb-node[data-name="${e.objectName}"] ueb-pin[data-id="${e.pinGuid}"]`)}getCopyInputObject(){return this.getInputObject(je)}isPointVisible(e,t){return!1}gridTopVisibilityBoundary(){return this.blueprint.scaleCorrect(this.blueprint.scrollY)-this.blueprint.translateY}gridRightVisibilityBoundary(){return this.blueprint,this.gridLeftVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[0])}gridBottomVisibilityBoundary(){return this.gridTopVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[1])}gridLeftVisibilityBoundary(){return this.blueprint.scaleCorrect(this.blueprint.scrollX)-this.blueprint.translateX}centerViewport(e=0,t=0,i=!0){const n=this.gridLeftVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[0]/2),s=this.gridTopVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[1]/2);this.blueprint.scrollDelta(this.blueprint.scaleCorrectReverse(e-n),this.blueprint.scaleCorrectReverse(t-s),i)}centerContentInViewport(e=!0){let t=0,i=0;const n=this.blueprint.getNodes();for(const e of n)t+=e.leftBoundary()+e.rightBoundary(),i+=e.topBoundary()+e.bottomBoundary();t=n.length>0?Math.round(t/(2*n.length)):0,i=n.length>0?Math.round(i/(2*n.length)):0,this.centerViewport(t,i,e)}}class ut extends Qe{static properties={...super.properties,fromX:{type:Number,attribute:!1},fromY:{type:Number,attribute:!1},toX:{type:Number,attribute:!1},toY:{type:Number,attribute:!1}};constructor(){super(),this.fromX=0,this.fromY=0,this.toX=0,this.toY=0}setBothLocations([e,t]){this.fromX=e,this.fromY=t,this.toX=e,this.toY=t}addSourceLocation(e,t){this.fromX+=e,this.fromY+=t}addDestinationLocation(e,t){this.toX+=e,this.toY+=t}}class ct extends Ue{update(e){super.update(e);const[t,i,n,s]=[Math.round(this.element.fromX),Math.round(this.element.fromY),Math.round(this.element.toX),Math.round(this.element.toY)],[r,a,o,l]=[Math.min(t,n),Math.min(i,s),Math.abs(t-n),Math.abs(i-s)];(e.has("fromX")||e.has("toX"))&&(this.element.style.left=`${r}px`,this.element.style.width=`${o}px`),(e.has("fromY")||e.has("toY"))&&(this.element.style.top=`${a}px`,this.element.style.height=`${l}px`)}}class ht extends He{constructor(e={},t){super(e,!0),this.Class=new pe("/Script/BlueprintGraph.K2Node_Knot"),this.Name="K2Node_Knot";const i=new Be({PinName:"InputPin"},!0),n=new Be({PinName:"OutputPin",Direction:"EGPD_Output"},!0);t&&(i.copyTypeFrom(t),n.copyTypeFrom(t)),this.CustomProperties=[i,n]}}class dt extends Xe{static ignoreDbClick=e=>{};#$=e=>{this.options.strictTarget&&e.target!==e.currentTarget||(this.options.consumeEvent&&e.stopImmediatePropagation(),this.clickedPosition=this.locationFromEvent(e),this.blueprint.mousePosition[0]=this.clickedPosition[0],this.blueprint.mousePosition[1]=this.clickedPosition[1],this.dbclicked(this.clickedPosition))};#H;get onDbClick(){return this.#H}set onDbClick(e){this.#H=e}clickedPosition=[0,0];constructor(e,t,i={},n=dt.ignoreDbClick){i.consumeEvent??=!0,i.strictTarget??=!1,super(e,t,i),this.#H=n,this.listenEvents()}listenEvents(){this.target.addEventListener("dblclick",this.#$)}unlistenEvents(){this.target.removeEventListener("dblclick",this.#$)}dbclicked(e){this.onDbClick(e)}}class pt extends ct{static decreasingValue(e,t){const i=-e*t[0]**2,n=t[1]-i/t[0];return e=>i/e+n}static clampedLine(e,t){if(e[0]>t[0]){const i=e;e=t,t=i}const i=(t[1]-e[1])/(t[0]-e[0]),n=e[1]-i*e[0];return s=>st[0]?t[1]:i*s+n}static c1DecreasingValue=pt.decreasingValue(-.15,[100,15]);static c2DecreasingValue=pt.decreasingValue(-.05,[500,130]);static c2Clamped=pt.clampedLine([0,80],[200,40]);#O=`ueb-id-${Math.floor(1e12*Math.random())}`;#V=e=>{const t=new ht({},this.element.sourcePin.entity),i=st.getConstructor("ueb-node").newObject(t);i.setLocation(...this.blueprint.snapToGrid(...e));const n=i.template;this.blueprint.addGraphElement(i);const s=this.element.getInputPin(),r=this.element.getOutputPin();this.element.sourcePin=null,this.element.destinationPin=null;const a=st.getConstructor("ueb-link").newObject(r,n.inputPin);this.blueprint.addGraphElement(a),this.element.sourcePin=n.outputPin,this.element.destinationPin=s};createInputObjects(){return[...super.createInputObjects(),new dt(this.element.querySelector(".ueb-link-area"),this.blueprint,void 0,(e=>{e[0]+=te.knotOffset[0],e[1]+=te.knotOffset[1],this.#V(e)}))]}willUpdate(e){super.willUpdate(e);const t=this.element.sourcePin,i=this.element.destinationPin;if(e.has("fromX")||e.has("toX")){const e=this.element.fromX,n=this.element.toX,s=t?.nodeElement.getType()==te.nodeType.knot,r=i?.nodeElement.getType()==te.nodeType.knot;!s||i&&!r||(t?.isInput()&&n>e+te.distanceThreshold?this.element.sourcePin=t.nodeElement.template.outputPin:t?.isOutput()&&ne+te.distanceThreshold&&(this.element.destinationPin=i.nodeElement.template.inputPin))}const n=Math.max(Math.abs(this.element.fromX-this.element.toX),1),s=Math.max(Math.abs(this.element.fromY-this.element.toY),1),r=Math.max(n,te.linkMinWidth),a=n/r,o=this.element.originatesFromInput?this.element.fromXthis.element.toY?1:0)),this.element.style.setProperty("--ueb-start-percentage",`${Math.round(this.element.startPercentage)}%`),this.element.style.setProperty("--ueb-link-start",`${Math.round(this.element.startPixels)}`)}render(){return $` ${this.element.linkMessageIcon||this.element.linkMessageText?$``:O}`}}class mt extends ut{static properties={...super.properties,source:{type:String,reflect:!0},destination:{type:String,reflect:!0},dragging:{type:Boolean,attribute:"data-dragging",converter:ae.booleanConverter,reflect:!0},originatesFromInput:{type:Boolean,attribute:!1},svgPathD:{type:String,attribute:!1},linkMessageIcon:{type:String,attribute:!1},linkMessageText:{type:String,attribute:!1}};#z;get sourcePin(){return this.#z}set sourcePin(e){this.#I(e,!1)}#R;get destinationPin(){return this.#R}set destinationPin(e){this.#I(e,!0)}#_=()=>this.remove();#F=e=>this.addSourceLocation(...e.detail.value);#G=e=>this.addDestinationLocation(...e.detail.value);#j=e=>this.setSourceLocation();#U=e=>this.setDestinationLocation();linkMessageIcon=O;linkMessageText=O;pathElement;constructor(){super(),this.source=null,this.destination=null,this.dragging=!1,this.originatesFromInput=!1,this.startPercentage=0,this.svgPathD="",this.startPixels=0}static newObject(e,t){const i=new mt;return i.initialize(e,t),i}initialize(e,t){super.initialize({},new pt),e&&(this.sourcePin=e,t||(this.toX=this.fromX,this.toY=this.fromY)),t&&(this.destinationPin=t,e||(this.fromX=this.toX,this.fromY=this.toY))}#I(e,t){const i=()=>t?this.destinationPin:this.sourcePin;if(i()!=e){if(i()){const e=i().getNodeElement();e.removeEventListener(te.removeEventName,this.#_),e.removeEventListener(te.nodeDragEventName,t?this.#G:this.#F),e.removeEventListener(te.nodeReflowEventName,t?this.#U:this.#j),this.#K()}if(t?this.#R=e:this.#z=e,i()){const e=i().getNodeElement();e.addEventListener(te.removeEventName,this.#_),e.addEventListener(te.nodeDragEventName,t?this.#G:this.#F),e.addEventListener(te.nodeReflowEventName,t?this.#U:this.#j),t?this.setDestinationLocation():(this.setSourceLocation(),this.originatesFromInput=this.sourcePin.isInput()),this.#W()}}}#W(){this.sourcePin&&this.destinationPin&&(this.sourcePin.linkTo(this.destinationPin),this.destinationPin.linkTo(this.sourcePin))}#K(){this.sourcePin&&this.destinationPin&&(this.sourcePin.unlinkFrom(this.destinationPin,!1),this.destinationPin.unlinkFrom(this.sourcePin,!1))}cleanup(){super.cleanup(),this.#K(),this.sourcePin=null,this.destinationPin=null}setSourceLocation(e=null,t=!0){if(null==e){const i=this;if(t&&(!this.hasUpdated||!this.sourcePin.hasUpdated))return void Promise.all([this.updateComplete,this.sourcePin.updateComplete]).then((()=>i.setSourceLocation(null,!1)));e=this.sourcePin.template.getLinkLocation()}const[i,n]=e;this.fromX=i,this.fromY=n}setDestinationLocation(e=null,t=!0){if(null==e){const i=this;if(t&&(!this.hasUpdated||!this.destinationPin.hasUpdated))return void Promise.all([this.updateComplete,this.destinationPin.updateComplete]).then((()=>i.setDestinationLocation(null,!1)));e=this.destinationPin.template.getLinkLocation()}this.toX=e[0],this.toY=e[1]}getInputPin(){return this.sourcePin?.isInput()?this.sourcePin:this.destinationPin}setInputPin(e){this.sourcePin?.isInput()&&(this.sourcePin=e),this.destinationPin=e}getOutputPin(){return this.destinationPin?.isOutput()?this.destinationPin:this.sourcePin}setOutputPin(e){this.destinationPin?.isOutput()&&(this.destinationPin=e),this.sourcePin=e}startDragging(){this.dragging=!0}finishDragging(){this.dragging=!1}removeMessage(){this.linkMessageIcon=O,this.linkMessageText=O}setMessageConvertType(){this.linkMessageIcon="ueb-icon-conver-type",this.linkMessageText=`Convert ${this.sourcePin.pinType} to ${this.destinationPin.pinType}.`}setMessageCorrect(){this.linkMessageIcon=De.correct,this.linkMessageText=O}setMessageReplace(){this.linkMessageIcon=De.correct,this.linkMessageText=O}setMessageDirectionsIncompatible(){this.linkMessageIcon=De.reject,this.linkMessageText=$`Directions are not compatbile.`}setMessagePlaceNode(){this.linkMessageIcon="ueb-icon-place-node",this.linkMessageText=$`Place a new node.`}setMessageReplaceLink(){this.linkMessageIcon=De.correct,this.linkMessageText=$`Replace existing input connections.`}setMessageReplaceOutputLink(){this.linkMessageIcon=De.correct,this.linkMessageText=$`Replace existing output connections.`}setMessageSameNode(){this.linkMessageIcon=De.reject,this.linkMessageText=$`Both are on the same node.`}setMEssagetypesIncompatible(){this.linkMessageIcon=De.reject,this.linkMessageText=$`${this.sourcePin.pinType} is not compatible with ${this.destinationPin.pinType}.`}}class gt extends tt{clicked(e){this.options.repositionOnClick&&(this.target.setLocation(...this.stepSize>1?ae.snapToGrid(e[0],e[1],this.stepSize):e),this.clickedOffset=[0,0])}dragTo(e,t){const i=[this.target.locationX??this.lastLocation[0],this.target.locationY??this.lastLocation[1]],[n,s]=this.stepSize>1?[ae.snapToGrid(e[0],e[1],this.stepSize),ae.snapToGrid(i[0],i[1],this.stepSize)]:[e,i];0==(t=[n[0]-this.lastLocation[0],n[1]-this.lastLocation[1]])[0]&&0==t[1]||(t[0]+=s[0]-i[0],t[1]+=s[1]-i[1],this.dragAction(n,t),this.lastLocation=n)}dragAction(e,t){this.target.setLocation(e[0]-this.clickedOffset[0],e[1]-this.clickedOffset[1])}}class bt extends gt{#X;#Z;#Y;#q;constructor(e,t,i={}){super(e,t,i),i.onClicked&&(this.#X=i.onClicked),i.onStartDrag&&(this.#Z=i.onStartDrag),i.onDrag&&(this.#Y=i.onDrag),i.onEndDrag&&(this.#q=i.onEndDrag)}clicked(e){super.clicked(e),this.#X?.()}startDrag(){super.startDrag(),this.#Z?.()}dragAction(e,t){this.#Y?.(e,t)}endDrag(){super.endDrag(),this.#q?.()}}class vt extends Ue{getDraggableElement(){return this.element}createDraggableObject(){return new gt(this.element,this.blueprint,{draggableElement:this.getDraggableElement()})}createInputObjects(){return[...super.createInputObjects(),this.createDraggableObject()]}topBoundary(e=!1){return this.element.locationY}rightBoundary(e=!1){return this.element.locationX+this.element.sizeX}bottomBoundary(e=!1){return this.element.locationY+this.element.sizeY}leftBoundary(e=!1){return this.element.locationX}centerInViewport(){const e=Math.min(this.blueprint.template.viewportSize[0]/10,this.blueprint.template.viewportSize[1]/10),t=this.leftBoundary()-this.blueprint.template.gridLeftVisibilityBoundary(),i=this.blueprint.template.gridRightVisibilityBoundary()-this.rightBoundary();let n=Math.max((t+i)/2,e);const s=this.topBoundary()-this.blueprint.template.gridTopVisibilityBoundary(),r=this.blueprint.template.gridBottomVisibilityBoundary()-this.bottomBoundary();let a=Math.max((s+r)/2,e);this.blueprint.scrollDelta(t-n,s-a,!0)}}class ft extends vt{update(e){super.update(e),e.has("locationX")&&(this.element.style.left=`${this.element.locationX}px`),e.has("locationY")&&(this.element.style.top=`${this.element.locationY}px`)}}class yt extends gt{startDrag(){this.target.selected||(this.blueprint.unselectAll(),this.target.setSelected(!0))}dragAction(e,t){this.target.acknowledgeDrag(t)}unclicked(){this.started?(this.blueprint.getNodes(!0).forEach((e=>e.boundComments.filter((t=>!e.isInsideComment(t))).forEach((t=>e.unbindFromComment(t))))),this.blueprint.getCommentNodes().forEach((e=>e.template.manageNodesBind()))):(this.blueprint.unselectAll(),this.target.setSelected(!0))}}class wt extends ft{getDraggableElement(){return this.element}createDraggableObject(){return new yt(this.element,this.blueprint,{draggableElement:this.getDraggableElement(),scrollGraphEdge:!0})}firstUpdated(e){super.firstUpdated(e),this.element.selected&&!this.element.listeningDrag&&this.element.setSelected(!0)}}class Et extends wt{hasSubtitle=!1;static nodeStyleClasses=["ueb-node-style-default"];toggleAdvancedDisplayHandler=()=>{this.element.toggleShowAdvancedPinDisplay(),this.element.addNextUpdatedCallbacks((()=>this.element.acknowledgeReflow()),!0)};initialize(e){super.initialize(e),this.element.classList.add(...this.constructor.nodeStyleClasses),this.element.style.setProperty("--ueb-node-color",this.getColor().cssText)}getColor(){return this.element.entity.nodeColor()}render(){return $`
${this.renderTop()}
${this.element.entity.isDevelopmentOnly()?$`
Development Only
`:O} ${this.element.advancedPinDisplay?$`
${De.expandIcon}
`:O}
`}renderNodeIcon(){return this.element.entity.nodeIcon()}renderNodeName(){return this.element.getNodeDisplayName()}renderTop(){const e=this.renderNodeIcon(),t=this.renderNodeName();return $`
${e?$`
${e}
`:O} ${t?$`
${t} ${this.hasSubtitle&&this.getTargetType().length>0?$`
Target is ${ae.formatStringName(this.getTargetType())}
`:O}
`:O}
`}firstUpdated(e){super.firstUpdated(e),this.setupPins(),this.element.updateComplete.then((()=>this.element.acknowledgeReflow()))}setupPins(){const e=this.element.querySelector(".ueb-node-inputs"),t=this.element.querySelector(".ueb-node-outputs");this.element.nodeNameElement=this.element.querySelector(".ueb-node-name-text");let i=!1,n=!1;this.element.getPinElements().forEach((s=>{s.isInput()?(e.appendChild(s),i=!0):s.isOutput()&&(t.appendChild(s),n=!0)})),i&&this.element.classList.add("ueb-node-has-inputs"),n&&this.element.classList.add("ueb-node-has-outputs")}createPinElements(){return this.element.getPinEntities().filter((e=>!e.isHidden())).map((e=>(this.hasSubtitle=this.hasSubtitle||"self"===e.PinName&&"Target"===e.getDisplayName(),st.getConstructor("ueb-pin").newObject(e,void 0,this.element))))}getTargetType(){return this.element.entity.FunctionReference?.MemberParent?.getName()??"Untitled"}getPinElements(e){return e.querySelectorAll("ueb-pin")}linksChanged(){}}class Ct extends Et{#J=document.createElement("div");#Q=document.createElement("div");#ee=document.createElement("div");#te=document.createElement("div");#ie=document.createElement("div");#ne=document.createElement("div");#se=document.createElement("div");#re=document.createElement("div");initialize(e){super.initialize(e),this.element.classList.add("ueb-resizeable"),this.#J.classList.add("ueb-resizeable-top"),this.#Q.classList.add("ueb-resizeable-right"),this.#ee.classList.add("ueb-resizeable-bottom"),this.#te.classList.add("ueb-resizeable-left"),this.#ie.classList.add("ueb-resizeable-top-right"),this.#ne.classList.add("ueb-resizeable-bottom-right"),this.#se.classList.add("ueb-resizeable-bottom-left"),this.#re.classList.add("ueb-resizeable-top-left")}update(e){super.update(e),this.element.sizeX>=0&&e.has("sizeX")&&(this.element.style.width=`${this.element.sizeX}px`),this.element.sizeY>=0&&e.has("sizeY")&&(this.element.style.height=`${this.element.sizeY}px`)}firstUpdated(e){super.firstUpdated(e),this.element.append(this.#J,this.#Q,this.#ee,this.#te,this.#ie,this.#ne,this.#se,this.#re)}createInputObjects(){return[...super.createInputObjects(),new bt(this.#J,this.blueprint,{onDrag:(e,t)=>{t[1]=e[1]-this.element.topBoundary(),this.setSizeY(this.element.sizeY-t[1])&&this.element.addLocation(0,t[1],!1)},onEndDrag:()=>this.endResize()}),new bt(this.#Q,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.rightBoundary(),this.setSizeX(this.element.sizeX+t[0])},onEndDrag:()=>this.endResize()}),new bt(this.#ee,this.blueprint,{onDrag:(e,t)=>{t[1]=e[1]-this.element.bottomBoundary(),this.setSizeY(this.element.sizeY+t[1])},onEndDrag:()=>this.endResize()}),new bt(this.#te,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.leftBoundary(),this.setSizeX(this.element.sizeX-t[0])&&this.element.addLocation(t[0],0,!1)},onEndDrag:()=>this.endResize()}),new bt(this.#ie,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.rightBoundary(),t[1]=e[1]-this.element.topBoundary(),this.setSizeX(this.element.sizeX+t[0]),this.setSizeY(this.element.sizeY-t[1])&&this.element.addLocation(0,t[1],!1)},onEndDrag:()=>this.endResize()}),new bt(this.#ne,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.rightBoundary(),t[1]=e[1]-this.element.bottomBoundary(),this.setSizeX(this.element.sizeX+t[0]),this.setSizeY(this.element.sizeY+t[1])},onEndDrag:()=>this.endResize()}),new bt(this.#se,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.leftBoundary(),t[1]=e[1]-this.element.bottomBoundary(),this.setSizeX(this.element.sizeX-t[0])&&this.element.addLocation(t[0],0,!1),this.setSizeY(this.element.sizeY+t[1])},onEndDrag:()=>this.endResize()}),new bt(this.#re,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.leftBoundary(),t[1]=e[1]-this.element.topBoundary(),this.setSizeX(this.element.sizeX-t[0])&&this.element.addLocation(t[0],0,!1),this.setSizeY(this.element.sizeY-t[1])&&this.element.addLocation(0,t[1],!1)},onEndDrag:()=>this.endResize()})]}setSizeX(e){return this.element.setNodeWidth(e),!0}setSizeY(e){return this.element.setNodeHeight(e),!0}endResize(){}}class St extends Ct{#ae=we.getWhite();#oe=0;initialize(e){e.entity.CommentColor&&(this.#ae.setFromRGBANumber(e.entity.CommentColor.toNumber()),this.#ae.setFromHSVA(this.#ae.H.value,this.#ae.S.value,.67*Math.pow(this.#ae.V.value,.45))),e.classList.add("ueb-node-style-comment","ueb-node-resizeable"),e.sizeX=25*te.gridSize,e.sizeY=6*te.gridSize,super.initialize(e)}getColor(){return r`${Math.round(255*this.#ae.R.value)}${Math.round(255*this.#ae.G.value)}${Math.round(255*this.#ae.B.value)}`}getDraggableElement(){return this.element.querySelector(".ueb-node-top")}render(){return $`
${this.element.entity.NodeComment}
`}firstUpdated(e){super.firstUpdated(e);const t=this.getDraggableElement().getBoundingClientRect();this.#oe=t.height}manageNodesBind(){let e=this.blueprint.getNodes();for(let t of e)t.topBoundary()>=this.element.topBoundary()&&t.rightBoundary()<=this.element.rightBoundary()&&t.bottomBoundary()<=this.element.bottomBoundary()&&t.leftBoundary()>=this.element.leftBoundary()?t.bindToComment(this.element):t.unbindFromComment(this.element)}setSizeX(e){return(e=Math.round(e))>=te.gridSet*te.gridSize&&(this.element.setNodeWidth(e),!0)}setSizeY(e){return(e=Math.round(e))>=3*te.gridSize&&(this.element.setNodeHeight(e),!0)}endResize(){this.manageNodesBind()}topBoundary(e=!1){return this.element.locationY}rightBoundary(e=!1){return this.element.locationX+this.element.sizeX}bottomBoundary(e=!1){return e?this.element.locationY+this.#oe:super.bottomBoundary()}leftBoundary(e=!1){return this.element.locationX}}class Pt extends tt{#le;#ue=null;#ce=e=>{if(!this.enteredPin){this.linkValid=!1,this.enteredPin=e.target;const t=this.link.sourcePin??this.target,i=this.enteredPin,n=t.isOutput()?t:i;t.nodeElement.getType()===te.nodeType.knot||i.nodeElement.getType()===te.nodeType.knot?(this.link.setMessageCorrect(),this.linkValid=!0):t.getNodeElement()===i.getNodeElement()?this.link.setMessageSameNode():t.isOutput()===i.isOutput()?this.link.setMessageDirectionsIncompatible():this.blueprint.getLinks(t,i).length?(this.link.setMessageReplaceLink(),this.linkValid=!0):"exec"===n.entity.getType()&&n.isLinked?(this.link.setMessageReplaceOutputLink(),this.linkValid=!0):(this.link.setMessageCorrect(),this.linkValid=!0)}};#he=e=>{this.enteredPin==e.target&&(this.enteredPin=null,this.linkValid=!1,this.link?.setMessagePlaceNode())};link;enteredPin;linkValid=!1;constructor(e,t,i={}){i.scrollGraphEdge??=!0,super(e,t,i)}startDrag(e){this.target.nodeElement.getType()==te.nodeType.knot&&(this.#ue=this.target),this.link=st.getConstructor("ueb-link").newObject(this.target,null),this.blueprint.template.linksContainerElement.prepend(this.link),this.link.setMessagePlaceNode(),this.#le=this.blueprint.querySelectorAll("ueb-pin"),this.#le.forEach((e=>{e!=this.target&&(e.addEventListener("mouseenter",this.#ce),e.addEventListener("mouseleave",this.#he))})),this.link.startDragging(),this.link.setDestinationLocation(e)}dragTo(e,t){this.link.setDestinationLocation(e)}endDrag(){if(this.#le.forEach((e=>{e.removeEventListener("mouseenter",this.#ce),e.removeEventListener("mouseleave",this.#he)})),this.enteredPin&&this.linkValid){if(this.#ue){const e=this.#ue!==this.link.sourcePin?this.link.sourcePin:this.enteredPin;if(this.#ue.isInput()&&e.isInput()||this.#ue.isOutput()&&e.isOutput()){const e=this.#ue.template.getOppositePin();this.#ue===this.link.sourcePin?this.link.sourcePin=e:this.enteredPin=e}}else this.enteredPin.nodeElement.getType()===te.nodeType.knot&&(this.enteredPin=this.enteredPin.template.getOppositePin());this.blueprint.addGraphElement(this.link),this.link.destinationPin=this.enteredPin,this.link.removeMessage(),this.link.finishDragging()}else this.link.finishDragging(),this.link.remove();this.enteredPin=null,this.link=null,this.#le=null}}class xt extends Et{#de=!1;#pe=!1;#me="";static nodeStyleClasses=["ueb-node-style-glass"];initialize(e){super.initialize(e),this.#me=this.element.getNodeDisplayName()}render(){return $`
${this.#me?$`
${this.#me}
`:O}
${this.#de?$`
`:O} ${this.#pe?$`
`:O}
`}createPinElements(){return this.element.getPinEntities().filter((e=>!e.isHidden())).map((e=>{this.#de||=e.isInput(),this.#pe||=e.isOutput();return st.getConstructor("ueb-pin").newObject(e,void 0,this.element)}))}}class kt extends xt{static nodeStyleClasses=[...super.nodeStyleClasses,"ueb-node-style-conversion"]}class Lt extends xt{static nodeStyleClasses=[...super.nodeStyleClasses,"ueb-node-style-operation"]}class Nt extends Ue{static canWrapInput=!0;#ge;get iconElement(){return this.#ge}#be;get wrapperElement(){return this.#be}isNameRendered=!0;setup(){super.setup(),this.element.nodeElement=this.element.closest("ueb-node");const e=this.element.nodeElement.template;(e instanceof kt||e instanceof Lt)&&(this.isNameRendered=!1,this.element.requestUpdate())}createInputObjects(){return[new Pt(this.element,this.blueprint,{moveEverywhere:!0,draggableElement:this.#be})]}render(){const e=$`
${this.renderIcon()}
`,t=$`
${this.isNameRendered?this.renderName():O} ${this.element.isInput()&&!this.element.entity.bDefaultValueIsIgnored?this.renderInput():$``}
`;return $`
${this.element.isInput()?$`${e}${t}`:$`${t}${e}`}
`}renderIcon(){switch(this.element.entity.PinType.ContainerType.toString()){case"Array":return De.array;case"Set":return De.set;case"Map":return De.map}return"delegate"===this.element.entity.PinType.PinCategory?De.delegate:De.genericPin}renderName(){return $`${this.element.getPinDisplayName()}`}renderInput(){return $``}updated(e){if(super.updated(e),this.element.isInput()&&e.has("isLinked")){const e=this.element.nodeElement;e.addNextUpdatedCallbacks((()=>e.acknowledgeReflow())),e.requestUpdate()}}firstUpdated(e){super.firstUpdated(e),this.element.style.setProperty("--ueb-pin-color-rgb",this.element.entity.pinColor().cssText),this.#ge=this.element.querySelector(".ueb-pin-icon svg")??this.element,this.#be=this.element.querySelector(".ueb-pin-wrapper")}getLinkLocation(){const e=this.iconElement.getBoundingClientRect(),t=[this.element.isInput()?e.left:e.right,(e.top+e.bottom)/2],i=ae.convertLocation(t,this.blueprint.template.gridElement);return this.blueprint.compensateTranslation(i[0],i[1])}getClickableElement(){return this.#be??this.element}}class Tt extends Nt{render(){return $`
${this.renderIcon()}
`}}class At extends Et{static nodeStyleClasses=[...super.nodeStyleClasses,"ueb-node-style-event"];firstUpdated(e){super.firstUpdated(e),this.element.querySelector(".ueb-node-top").appendChild(this.createDelegatePinElement())}renderTop(){const e=this.renderNodeIcon(),t=this.renderNodeName();return $`
${e?$`
${e}
`:O} ${t?$`
${t} ${this.hasSubtitle&&this.element.entity.FunctionReference.MemberParent?$`
Custom Event
`:O}
`:O}
`}createDelegatePinElement(){const e=st.getConstructor("ueb-pin").newObject(this.element.getPinEntities().find((e=>!e.isHidden()&&"delegate"===e.PinType.PinCategory)),new Tt,this.element);return e.template.isNameRendered=!1,e}createPinElements(){return this.element.getPinEntities().filter((e=>!e.isHidden()&&"delegate"!==e.PinType.PinCategory)).map((e=>st.getConstructor("ueb-pin").newObject(e,void 0,this.element)))}}class Mt extends et{static properties={...super.properties,selected:{type:Boolean,attribute:"data-selected",reflect:!0,converter:ae.booleanConverter}};dragHandler=e=>this.addLocation(...e.detail.value);constructor(){super(),this.selected=!1,this.listeningDrag=!1}setup(){super.setup(),this.setSelected(this.selected)}cleanup(){super.cleanup(),this.blueprint.removeEventListener(te.nodeDragGeneralEventName,this.dragHandler)}setSelected(e=!0){this.selected=e,this.blueprint&&(this.selected?(this.listeningDrag=!0,this.blueprint.addEventListener(te.nodeDragGeneralEventName,this.dragHandler)):(this.blueprint.removeEventListener(te.nodeDragGeneralEventName,this.dragHandler),this.listeningDrag=!1))}}class Bt extends Tt{render(){return this.element.isOutput()?super.render():$``}getOppositePin(){const e=this.element.nodeElement.template;return this.element.isOutput()?e.inputPin:e.outputPin}getLinkLocation(){const e=(this.element.isInput()?this.element.nodeElement.template.outputPin.template:this).iconElement.getBoundingClientRect(),t=[this.element.isInput()?e.left:e.right,(e.top+e.bottom)/2],i=ae.convertLocation(t,this.blueprint.template.gridElement);return this.blueprint.compensateTranslation(i[0],i[1])}}class Dt extends Et{static#ve=new Set;#fe=null;#ye;get inputPin(){return this.#ye}#we;get outputPin(){return this.#we}initialize(e){super.initialize(e),this.element.classList.add("ueb-node-style-minimal")}findDirectionaPin(e){if(e.nodeElement.getType()!==te.nodeType.knot||Dt.#ve.has(e))return Dt.#ve.clear(),!0;Dt.#ve.add(e);for(let t of e.getLinks().map((e=>this.blueprint.getPin(e))))if(this.findDirectionaPin(t))return!0;return!1}render(){return $`
`}setupPins(){this.element.getPinElements().forEach((e=>this.element.querySelector(".ueb-node-border").appendChild(e)))}getPinElements(e){return e.querySelectorAll("ueb-pin")}createPinElements(){const e=this.element.getPinEntities().filter((e=>!e.isHidden())),t=e[e[0].isInput()?0:1],i=e[e[0].isOutput()?0:1],n=st.getConstructor("ueb-pin");return[this.#ye=n.newObject(t,new Bt,this.element),this.#we=n.newObject(i,new Bt,this.element)]}linksChanged(){}}class $t extends xt{initialize(e){super.initialize(e),e.getType()===te.nodeType.variableGet?this.element.classList.add("ueb-node-style-getter"):e.getType()===te.nodeType.variableSet&&this.element.classList.add("ueb-node-style-setter")}setupPins(){super.setupPins();let e=this.element.getPinElements().find((e=>!e.entity.isHidden()&&!e.entity.isExecution()));this.element.style.setProperty("--ueb-node-color",e.getColor().cssText)}}class Ht extends Mt{static properties={...Mt.properties,typePath:{type:String,attribute:"data-type",reflect:!0},nodeName:{type:String,attribute:"data-name",reflect:!0},advancedPinDisplay:{type:String,attribute:"data-advanced-display",converter:ge.attributeConverter,reflect:!0},enabledState:{type:String,attribute:"data-enabled-state",reflect:!0},nodeDisplayName:{type:String,attribute:!1},pureFunction:{type:Boolean,converter:ae.booleanConverter,attribute:"data-pure-function",reflect:!0}};static dragEventName=te.nodeDragEventName;static dragGeneralEventName=te.nodeDragGeneralEventName;get blueprint(){return super.blueprint}set blueprint(e){super.blueprint=e,this.#Ee.forEach((t=>t.blueprint=e))}#Ce;get nodeNameElement(){return this.#Ce}set nodeNameElement(e){this.#Ce=e}#Ee=[];boundComments=[];#Se=!1;#Pe=e=>{this.selected||this.#Se||(this.#Se=!0,this.addNextUpdatedCallbacks((()=>this.#Se=!1)),this.addLocation(...e.detail.value))};static getTypeTemplate(e){if(e.getClass()===te.nodeType.callFunction||e.getClass()===te.nodeType.commutativeAssociativeBinaryOperator||e.getClass()===te.nodeType.callArrayFunction){const t=e.FunctionReference.MemberParent?.path??"";if("/Script/Engine.KismetMathLibrary"===t||"/Script/Engine.KismetArrayLibrary"===t){if(e.FunctionReference.MemberName?.startsWith("Conv_"))return kt;if(e.FunctionReference.MemberName?.startsWith("Percent_"))return Lt;switch(e.FunctionReference.MemberName){case"Array_Add":case"Array_Identical":case"Abs":case"Array_Add":case"BMax":case"BMin":case"Exp":case"FMax":case"FMin":case"Max":case"MaxInt64":case"Min":case"MinInt64":return Lt}}if("/Script/Engine.BlueprintSetLibrary"===t)return Lt;if("/Script/Engine.BlueprintMapLibrary"===t)return Lt}switch(e.getClass()){case te.nodeType.comment:return St;case te.nodeType.event:case te.nodeType.customEvent:return At;case te.nodeType.promotableOperator:return Lt;case te.nodeType.knot:return Dt;case te.nodeType.variableGet:case te.nodeType.variableSet:return $t}return e.getDelegatePin()?At:Et}static fromSerializedObject(e){e=e.trim();let t=ne.getSerializer(He).deserialize(e);return Ht.newObject(t)}static newObject(e=new He,t=new(Ht.getTypeTemplate(e))){const i=new Ht;return i.initialize(e,t),i}initialize(e=new He,t=new(Ht.getTypeTemplate(e))){super.initialize(e,t),this.#Ee=this.template.createPinElements(),this.typePath=this.entity.getType(),this.nodeName=this.entity.getObjectName(),this.advancedPinDisplay=this.entity.AdvancedPinDisplay?.toString(),this.enabledState=this.entity.EnabledState,this.nodeDisplayName=this.getNodeDisplayName(),this.pureFunction=this.entity.bIsPureFunc,this.dragLinkObjects=[],super.setLocation(this.entity.getNodePosX(),this.entity.getNodePosY()),this.entity.NodeWidth&&this.entity.NodeHeight?(this.sizeX=this.entity.NodeWidth.value,this.sizeY=this.entity.NodeHeight.value):this.updateComplete.then((()=>this.computeSizes()))}getUpdateComplete(){return Promise.all([super.getUpdateComplete(),...this.getPinElements().map((e=>e.updateComplete))]).then((()=>!0))}bindToComment(e){e==this||this.boundComments.includes(e)||(e.addEventListener(te.nodeDragEventName,this.#Pe),this.boundComments.push(e))}unbindFromComment(e){const t=this.boundComments.indexOf(e);t>=0&&(e.removeEventListener(te.nodeDragEventName,this.#Pe),this.boundComments[t]=this.boundComments[this.boundComments.length-1],this.boundComments.pop())}isInsideComment(e){return this.topBoundary()>=e.topBoundary()&&this.rightBoundary()<=e.rightBoundary()&&this.bottomBoundary()<=e.bottomBoundary()&&this.leftBoundary()>=e.leftBoundary()}getType(){return this.entity.getType()}getNodeName(){return this.entity.getObjectName()}getNodeDisplayName(){return this.entity.nodeDisplayName()}setNodeWidth(e){this.entity.setNodeWidth(e),this.sizeX=e,this.acknowledgeReflow()}setNodeHeight(e){this.entity.setNodeHeight(e),this.sizeY=e,this.acknowledgeReflow()}sanitizeLinks(e=[]){this.getPinElements().forEach((t=>t.sanitizeLinks(e)))}rename(e){if(this.entity.Name==e)return!1;for(let t of this.getPinElements())for(let i of t.getLinks())this.blueprint.getPin(i).redirectLink(t,new Pe({objectName:e,pinGuid:t.entity.PinId}));this.entity.Name=e,this.nodeName=this.entity.Name}getPinElements(){return this.#Ee}getPinEntities(){return this.entity.CustomProperties.filter((e=>e instanceof Be))}setLocation(e=0,t=0,i=!0){this.entity.setNodePosX(e),this.entity.setNodePosY(t),super.setLocation(e,t,i)}acknowledgeReflow(){this.requestUpdate(),this.updateComplete.then((()=>this.computeSizes()));let e=new CustomEvent(te.nodeReflowEventName);this.dispatchEvent(e)}setShowAdvancedPinDisplay(e){this.entity.AdvancedPinDisplay=new ge(e?"Shown":"Hidden"),this.advancedPinDisplay=this.entity.AdvancedPinDisplay}toggleShowAdvancedPinDisplay(){this.setShowAdvancedPinDisplay("Shown"!=this.entity.AdvancedPinDisplay?.toString())}}class Ot extends Qe{static properties={selecting:{type:Boolean,attribute:"data-selecting",reflect:!0,converter:ae.booleanConverter},scrolling:{type:Boolean,attribute:"data-scrolling",reflect:!0,converter:ae.booleanConverter},focused:{type:Boolean,attribute:"data-focused",reflect:!0,converter:ae.booleanConverter},zoom:{type:Number,attribute:"data-zoom",reflect:!0},scrollX:{type:Number,attribute:!1},scrollY:{type:Number,attribute:!1},additionalX:{type:Number,attribute:!1},additionalY:{type:Number,attribute:!1},translateX:{type:Number,attribute:!1},translateY:{type:Number,attribute:!1}};static nodeBoundariesSupplier=e=>({primaryInf:e.leftBoundary(!0),primarySup:e.rightBoundary(!0),secondaryInf:e.topBoundary(!0),secondarySup:e.bottomBoundary(!0)});static nodeSelectToggleFunction=(e,t)=>{e.setSelected(t)};#xe=new Map;nodes=[];links=[];mousePosition=[0,0];waitingExpandUpdate=!1;constructor(){super(),this.selecting=!1,this.scrolling=!1,this.focused=!1,this.zoom=0,this.scrollX=te.expandGridSize,this.scrollY=te.expandGridSize,this.translateX=te.expandGridSize,this.translateY=te.expandGridSize,super.initialize({},new lt)}initialize(){}getGridDOMElement(){return this.template.gridElement}getScroll(){return[this.scrollX,this.scrollY]}setScroll(e,t){this.scrollX=e,this.scrollY=t}scrollDelta(e=0,t=0,i=!1,n=te.smoothScrollTime){if(i){let i=[0,0];ae.animate(0,e,n,(e=>{this.scrollDelta(e-i[0],0,!1),i[0]=e})),ae.animate(0,t,n,(e=>{this.scrollDelta(0,e-i[1],!1),i[1]=e}))}else{const i=[2*te.expandGridSize,2*te.expandGridSize];let n=this.getScroll(),s=[n[0]+e,n[1]+t],r=[0,0];for(let e=0;e<2;++e)s[e]i[e]-te.gridExpandThreshold*te.expandGridSize&&(r[e]=1);0==r[0]&&0==r[1]||this.seamlessExpand(r[0],r[1]),n=this.getScroll(),s=[n[0]+e,n[1]+t],this.setScroll(s[0],s[1])}}scrollCenter(){const e=this.getScroll(),t=[this.translateX-e[0],this.translateY-e[1]],i=this.getViewportSize().map((e=>e/2)),n=[t[0]-i[0],t[1]-i[1]];this.scrollDelta(n[0],n[1],!0)}getViewportSize(){return[this.template.viewportElement.clientWidth,this.template.viewportElement.clientHeight]}getScrollMax(){return[this.template.viewportElement.scrollWidth-this.template.viewportElement.clientWidth,this.template.viewportElement.scrollHeight-this.template.viewportElement.clientHeight]}snapToGrid(e,t){return ae.snapToGrid(e,t,te.gridSize)}seamlessExpand(e,t){e=Math.round(e),t=Math.round(t);let i=this.getScale();[e,t]=[-e*te.expandGridSize,-t*te.expandGridSize],0!=e&&(this.scrollX+=e,e/=i),0!=t&&(this.scrollY+=t,t/=i),this.translateX+=e,this.translateY+=t}progressiveSnapToGrid(e){return te.expandGridSize*Math.round(e/te.expandGridSize+.5*Math.sign(e))}getZoom(){return this.zoom}setZoom(e,t){if((e=ae.clamp(e,te.minZoom,te.maxZoom))==this.zoom)return;let i=this.getScale();if(this.zoom=e,t){t[0]+=this.translateX,t[1]+=this.translateY;let e=this.getScale()/i,n=[e*t[0],e*t[1]];this.scrollDelta((n[0]-t[0])*i,(n[1]-t[1])*i)}}getScale(){return te.scale[this.getZoom()]}scaleCorrect(e){return e/this.getScale()}scaleCorrectReverse(e){return e*this.getScale()}compensateTranslation(e,t){return[e-=this.translateX,t-=this.translateY]}getNodes(e=!1,[t,i,n,s]=[Number.MIN_SAFE_INTEGER,Number.MAX_SAFE_INTEGER,Number.MAX_SAFE_INTEGER,Number.MIN_SAFE_INTEGER]){let r=this.nodes;return e&&(r=r.filter((e=>e.selected))),(t>Number.MIN_SAFE_INTEGER||iNumber.MIN_SAFE_INTEGER)&&(r=r.filter((e=>e.topBoundary()>=t&&e.rightBoundary()<=i&&e.bottomBoundary()<=n&&e.leftBoundary()>=s))),r}getCommentNodes(e=!1){let t=[...this.template.getCommentNodes(e)];return 0===t.length&&(t=this.nodes.filter((t=>t.getType()===te.nodeType.comment&&(!e||t.selected)))),t}getPin(e){let t=this.template.getPin(e);return t&&t.nodeElement.getNodeName()==e.objectName.toString()||(t=[...this.nodes.find((t=>e.objectName.toString()==t.getNodeName()))?.getPinElements()??[]].find((t=>e.pinGuid.toString()==t.getPinId().toString()))),t}getLinks(e=null,t=null){if(null==e!=(null==t)){const i=e??t;return this.links.filter((e=>e.sourcePin==i||e.destinationPin==i))}return null!=e&&null!=t?this.links.filter((i=>i.sourcePin==e&&i.destinationPin==t||i.sourcePin==t&&i.destinationPin==e)):this.links}getLink(e,t,i=!1){return this.links.find((n=>n.sourcePin==e&&n.destinationPin==t||!i&&n.sourcePin==t&&n.destinationPin==e))}selectAll(){this.getNodes().forEach((e=>Ot.nodeSelectToggleFunction(e,!0)))}unselectAll(){this.getNodes().forEach((e=>Ot.nodeSelectToggleFunction(e,!1)))}addGraphElement(...e){const t=e=>{const i=e.currentTarget;i.removeEventListener(te.removeEventName,t);const n=i instanceof Ht?this.nodes:i instanceof mt?this.links:null,s=n?.indexOf(i);if(s>=0){const e=n.pop();st.entity.getObjectName()==e));if(n){let e=n.entity.getObjectName(!0);this.#xe[e]=this.#xe[e]??-1;do{++this.#xe[e]}while(this.nodes.find((t=>t.entity.getObjectName()==te.nodeName(e,this.#xe[e]))));n.rename(te.nodeName(e,this.#xe[e]))}this.nodes.push(i),i.addEventListener(te.removeEventName,t),this.template.nodesContainerElement?.appendChild(i)}else i instanceof mt&&!this.links.includes(i)&&(this.links.push(i),i.addEventListener(te.removeEventName,t),this.template.linksContainerElement&&!this.template.linksContainerElement.contains(i)&&this.template.linksContainerElement.appendChild(i));e.filter((e=>e instanceof Ht)).forEach((t=>t.sanitizeLinks(e))),e.filter((e=>e instanceof Ht&&e.getType()==te.nodeType.comment)).forEach((e=>e.updateComplete.then((()=>e.template.manageNodesBind()))))}removeGraphElement(...e){for(let t of e){if(t.closest("ueb-blueprint")!==this)return;t.remove()}}setFocused(e=!0){if(this.focused==e)return;let t=new CustomEvent(e?te.focusEventName.begin:te.focusEventName.end);this.focused=e,this.focused||this.unselectAll(),this.dispatchEvent(t)}acknowledgeEditText(e){const t=new CustomEvent(e?te.editTextEventName.begin:te.editTextEventName.end);this.dispatchEvent(t)}}customElements.define("ueb-blueprint",Ot);class Vt extends vt{#ke;get locationChangeCallback(){return this.#ke}set locationChangeCallback(e){this.#ke=e}movementSpace;movementSpaceSize=[0,0];firstUpdated(e){super.firstUpdated(e),this.movementSpace=this.element.parentElement}setup(){super.setup();const e=this.movementSpace.getBoundingClientRect();this.movementSpaceSize=[e.width,e.height]}createDraggableObject(){return new gt(this.element,this.blueprint,{draggableElement:this.movementSpace,ignoreTranslateCompensate:!0,moveEverywhere:!0,movementSpace:this.movementSpace,repositionOnClick:!0,stepSize:1})}adjustLocation(e,t){return this.locationChangeCallback?.(e,t),[e,t]}}class zt extends Vt{adjustLocation(e,t){const i=Math.round(this.movementSpaceSize[0]/2);e-=i,t=-(t-i);let[n,s]=ae.getPolarCoordinates(e,t);return n=Math.min(n,i),[e,t]=ae.getCartesianCoordinates(n,s),this.locationChangeCallback?.(e/i,t/i),[e=Math.round(e+i),t=Math.round(-t+i)]}}class It extends et{windowElement;setup(){super.setup(),this.windowElement=this.closest("ueb-window")}setLocation(e,t){super.setLocation(...this.template.adjustLocation(e,t))}}class Rt extends It{constructor(){super(),super.initialize({},new zt)}static newObject(){return new Rt}initialize(){}}class _t extends Vt{adjustLocation(e,t){return e=ae.clamp(e,0,this.movementSpaceSize[0]),t=ae.clamp(t,0,this.movementSpaceSize[1]),this.locationChangeCallback?.(e/this.movementSpaceSize[0],1-t/this.movementSpaceSize[1]),[e,t]}}class Ft extends It{constructor(){super(),super.initialize({},new _t)}static newObject(){return new Ft}initialize(){}}class Gt extends Ue{#Le=()=>{this.blueprint.acknowledgeEditText(!0),this.element.selectOnFocus&&getSelection().selectAllChildren(this.element)};#Ne=()=>{this.blueprint.acknowledgeEditText(!1),getSelection().removeAllRanges()};#Te=e=>e.target.querySelectorAll("br").forEach((e=>e.remove()));#Ae=e=>{"Enter"!=e.code||e.shiftKey||e.target.blur()};initialize(e){super.initialize(e),this.element.classList.add("ueb-pin-input-content"),this.element.setAttribute("role","textbox"),this.element.contentEditable="true"}setup(){super.setup(),this.element.addEventListener("focus",this.#Le),this.element.addEventListener("focusout",this.#Ne),this.element.singleLine&&this.element.addEventListener("input",this.#Te),this.element.blurOnEnter&&this.element.addEventListener("keydown",this.#Ae)}cleanup(){super.cleanup(),this.element.removeEventListener("focus",this.#Le),this.element.removeEventListener("focusout",this.#Ne),this.element.removeEventListener("input",this.#Te),this.element.removeEventListener("keydown",this.#Ae)}}class jt extends Qe{static properties={...super.properties,singleLine:{type:Boolean,attribute:"data-single-line",converter:ae.booleanConverter,reflect:!0},selectOnFocus:{type:Boolean,attribute:"data-select-focus",converter:ae.booleanConverter,reflect:!0},blurOnEnter:{type:Boolean,attribute:"data-blur-enter",converter:ae.booleanConverter,reflect:!0}};constructor(){super(),this.singleLine=!1,this.selectOnFocus=!0,this.blurOnEnter=!0,super.initialize({},new Gt)}static newObject(){return new jt}initialize(){}}class Ut extends tt{constructor(e,t,i={}){i.consumeEvent=!0,super(e,t,i)}}class Kt extends Nt{#Me;#Be=e=>this.element.setDefaultValue(this.#Me.checked);firstUpdated(e){super.firstUpdated(e),this.#Me=this.element.querySelector(".ueb-pin-input")}setup(){super.setup(),this.#Me?.addEventListener("change",this.#Be)}cleanup(){super.cleanup(),this.#Me?.removeEventListener("change",this.#Be)}createInputObjects(){return[...super.createInputObjects(),new Ut(this.#Me,this.blueprint)]}renderInput(){return $``}}class Wt extends Nt{renderIcon(){return De.execPin}renderName(){let e=this.element.entity.PinName;if(this.element.entity.PinFriendlyName)e=this.element.entity.PinFriendlyName.toString();else if("execute"===e||"then"===e)return $``;return $`${ae.formatStringName(e)}`}}class Xt extends Nt{static singleLineInput=!1;static selectOnFocus=!0;#De;get inputContentElements(){return this.#De}static stringFromInputToUE(e){return e.replace(/(?=\n\s*)\n$/,"").replaceAll("\n","\\r\n")}static stringFromUEToInput(e){return e.replaceAll(/(?:\r|(?<=(?:^|[^\\])(?:\\\\)*)\\r)(?=\n)/g,"").replace(/(?<=\n\s*)$/,"\n")}#$e=()=>this.setInputs(this.getInputs(),!0);#He=e=>this.#Oe(e.target);#Oe(e){const t=this.blueprint.scaleCorrect(e.getBoundingClientRect().width)+this.nameWidth,i=this.element.classList.contains("ueb-pin-input-wrap");!i&&t>te.pinInputWrapWidth?this.element.classList.add("ueb-pin-input-wrap"):i&&t<=te.pinInputWrapWidth&&this.element.classList.remove("ueb-pin-input-wrap")}firstUpdated(e){super.firstUpdated(e),this.#De=[...this.element.querySelectorAll("ueb-input")],this.constructor.canWrapInput&&(this.nameWidth=this.blueprint.scaleCorrect(this.element.querySelector(".ueb-pin-name").getBoundingClientRect().width),this.inputContentElements.forEach((e=>this.#Oe(e))))}setup(){super.setup(),this.#De.forEach((e=>{e.addEventListener("focusout",this.#$e),this.constructor.canWrapInput&&e.addEventListener("input",this.#He)}))}cleanup(){super.cleanup(),this.#De.forEach((e=>{e.removeEventListener("focusout",this.#$e),e.removeEventListener("input",this.#He)}))}createInputObjects(){return[...super.createInputObjects(),...this.#De.map((e=>new Ut(e,this.blueprint)))]}getInput(){return this.getInputs().reduce(((e,t)=>e+t),"")}getInputs(){return this.#De.map((e=>ae.clearHTMLWhitespace(e.innerHTML)))}setInputs(e=[],t=!0){this.#De.forEach(this.constructor.singleLineInput?(t,i)=>t.innerText=e[i]:(t,i)=>t.innerText=e[i].replaceAll("\n","")),t&&this.setDefaultValue(e.map((e=>Xt.stringFromInputToUE(e))),e),this.element.addNextUpdatedCallbacks((()=>this.element.nodeElement.acknowledgeReflow()))}setDefaultValue(e=[],t=e){this.element.setDefaultValue(e.join(""))}renderInput(){const e=this.constructor.singleLineInput,t=this.constructor.selectOnFocus;return $`
`}}class Zt extends Xt{static singleLineInput=!0;setInputs(e=[],t=!1){if(e&&0!=e.length||(e=[this.getInput()]),super.setInputs(e,!1),t){let i=[];for(const n of e){let e=parseFloat(n);isNaN(e)&&(e=0,t=!1),i.push(e)}this.setDefaultValue(i,e)}}setDefaultValue(e=[],t){this.element.setDefaultValue(e[0])}}class Yt extends Zt{setDefaultValue(e=[],t=e){this.element.getDefaultValue(!0).value=e[0],this.inputContentElements[0].innerText=this.element.getDefaultValue()?.toString(),this.element.requestUpdate()}renderInput(){return $`
`}}class qt extends Yt{setInputs(e=[],t=!1){if(e&&0!=e.length||(e=[this.getInput()]),super.setInputs(e,!1),t){if(!e[0].match(/[\-\+]?[0-9]+/))return;const t=[BigInt(e[0])];this.setDefaultValue(t,e)}}} +var q,J;class Q extends g{constructor(){super(...arguments),this.renderOptions={host:this},this._$Do=void 0}createRenderRoot(){var e,t;const i=super.createRenderRoot();return null!==(e=(t=this.renderOptions).renderBefore)&&void 0!==e||(t.renderBefore=i.firstChild),i}update(e){const t=this.render();this.hasUpdated||(this.renderOptions.isConnected=this.isConnected),super.update(e),this._$Do=((e,t,i)=>{var n,s;const r=null!==(n=null==i?void 0:i.renderBefore)&&void 0!==n?n:t;let a=r._$litPart$;if(void 0===a){const e=null!==(s=null==i?void 0:i.renderBefore)&&void 0!==s?s:null;r._$litPart$=a=new G(t.insertBefore(x(),e),e,void 0,null!=i?i:{})}return a._$AI(e),a})(t,this.renderRoot,this.renderOptions)}connectedCallback(){var e;super.connectedCallback(),null===(e=this._$Do)||void 0===e||e.setConnected(!0)}disconnectedCallback(){var e;super.disconnectedCallback(),null===(e=this._$Do)||void 0===e||e.setConnected(!1)}render(){return H}}Q.finalized=!0,Q._$litElement$=!0,null===(q=globalThis.litElementHydrateSupport)||void 0===q||q.call(globalThis,{LitElement:Q});const ee=globalThis.litElementPolyfillSupport;null==ee||ee({LitElement:Q}),(null!==(J=globalThis.litElementVersions)&&void 0!==J?J:globalThis.litElementVersions=[]).push("3.2.2");class te{static nodeColors={blue:r``,gray:r``,green:r``,red:r``,turquoise:r``};static alphaPattern="repeating-conic-gradient(#7c8184 0% 25%, #c2c3c4 0% 50%) 50% / 10px 10px";static colorDragEventName="ueb-color-drag";static colorPickEventName="ueb-color-pick";static colorWindowEventName="ueb-color-window";static colorWindowName="Color Picker";static defaultCommentHeight=96;static defaultCommentWidth=400;static deleteNodesKeyboardKey="Delete";static distanceThreshold=5;static dragEventName="ueb-drag";static dragGeneralEventName="ueb-drag-general";static edgeScrollThreshold=50;static editTextEventName={begin:"ueb-edit-text-begin",end:"ueb-edit-text-end"};static enableZoomIn=["LeftControl","RightControl"];static expandGridSize=400;static focusEventName={begin:"blueprint-focus",end:"blueprint-unfocus"};static fontSize=r``;static gridAxisLineColor=r``;static gridExpandThreshold=.25;static gridLineColor=r``;static gridLineWidth=1;static gridSet=8;static gridSetLineColor=r``;static gridShrinkThreshold=4;static gridSize=16;static hexColorRegex=/^\s*#(?[0-9a-fA-F]{2})(?[0-9a-fA-F]{2})(?[0-9a-fA-F]{2})([0-9a-fA-F]{2})?|#(?[0-9a-fA-F])(?[0-9a-fA-F])(?[0-9a-fA-F])\s*$/;static keysSeparator="+";static knotOffset=[-26,-16];static linkCurveHeight=15;static linkCurveWidth=80;static linkMinWidth=100;static linkRightSVGPath=(e,t,i)=>{let n=100-e;return`M ${e} 0 C ${t.toFixed(3)} 0, ${i.toFixed(3)} 0, 50 50 S ${(n-t+e).toFixed(3)} 100, ${n.toFixed(3)} 100`};static maxZoom=7;static minZoom=-12;static mouseWheelFactor=.2;static nodeDragGeneralEventName="ueb-node-drag-general";static nodeDragEventName="ueb-node-drag";static nodeName=(e,t)=>`${e}_${t}`;static nodeRadius=8;static nodeReflowEventName="ueb-node-reflow";static nodeType={callArrayFunction:"/Script/BlueprintGraph.K2Node_CallArrayFunction",callFunction:"/Script/BlueprintGraph.K2Node_CallFunction",comment:"/Script/UnrealEd.EdGraphNode_Comment",commutativeAssociativeBinaryOperator:"/Script/BlueprintGraph.K2Node_CommutativeAssociativeBinaryOperator",componentBoundEvent:"/Script/BlueprintGraph.K2Node_ComponentBoundEvent",customEvent:"/Script/BlueprintGraph.K2Node_CustomEvent",doN:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:Do N",doOnce:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:DoOnce",dynamicCast:"/Script/BlueprintGraph.K2Node_DynamicCast",enum:"/Script/CoreUObject.Enum",enumLiteral:"/Script/BlueprintGraph.K2Node_EnumLiteral",event:"/Script/BlueprintGraph.K2Node_Event",executionSequence:"/Script/BlueprintGraph.K2Node_ExecutionSequence",forEachElementInEnum:"/Script/BlueprintGraph.K2Node_ForEachElementInEnum",forEachLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForEachLoop",forEachLoopWithBreak:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForEachLoopWithBreak",forLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForLoop",forLoopWithBreak:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ForLoopWithBreak",functionEntry:"/Script/BlueprintGraph.K2Node_FunctionEntry",getInputAxisKeyValue:"/Script/BlueprintGraph.K2Node_GetInputAxisKeyValue",ifThenElse:"/Script/BlueprintGraph.K2Node_IfThenElse",inputAxisKeyEvent:"/Script/BlueprintGraph.K2Node_InputAxisKeyEvent",inputDebugKey:"/Script/InputBlueprintNodes.K2Node_InputDebugKey",inputKey:"/Script/BlueprintGraph.K2Node_InputKey",isValid:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:IsValid",knot:"/Script/BlueprintGraph.K2Node_Knot",macro:"/Script/BlueprintGraph.K2Node_MacroInstance",makeArray:"/Script/BlueprintGraph.K2Node_MakeArray",makeMap:"/Script/BlueprintGraph.K2Node_MakeMap",makeSet:"/Script/BlueprintGraph.K2Node_MakeSet",multiGate:"/Script/BlueprintGraph.K2Node_MultiGate",pawn:"/Script/Engine.Pawn",promotableOperator:"/Script/BlueprintGraph.K2Node_PromotableOperator",reverseForEachLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ReverseForEachLoop",select:"/Script/BlueprintGraph.K2Node_Select",userDefinedEnum:"/Script/Engine.UserDefinedEnum",variableGet:"/Script/BlueprintGraph.K2Node_VariableGet",variableSet:"/Script/BlueprintGraph.K2Node_VariableSet",whileLoop:"/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:WhileLoop"};static pinColor={"/Script/CoreUObject.Rotator":r``,"/Script/CoreUObject.Transform":r``,"/Script/CoreUObject.Vector":r``,bool:r``,byte:r``,class:r``,default:r``,delegate:r``,enum:r``,exec:r``,int:r``,int64:r``,interface:r``,name:r``,object:r``,real:r``,string:r``,struct:r``,text:r``,wildcard:r``};static pinInputWrapWidth=134;static removeEventName="ueb-element-delete";static scale={[-12]:.133333,[-11]:.166666,[-10]:.2,[-9]:.233333,[-8]:.266666,[-7]:.3,[-6]:.333333,[-5]:.375,[-4]:.5,[-3]:.675,[-2]:.75,[-1]:.875,0:1,1:1.25,2:1.375,3:1.5,4:1.675,5:1.75,6:1.875,7:2};static selectAllKeyboardKey="(bCtrl=True,Key=A)";static smoothScrollTime=1e3;static trackingMouseEventName={begin:"ueb-tracking-mouse-begin",end:"ueb-tracking-mouse-end"};static windowApplyEventName="ueb-window-apply";static windowApplyButtonText="OK";static windowCancelEventName="ueb-window-cancel";static windowCancelButtonText="Cancel";static windowCloseEventName="ueb-window-close";static ModifierKeys=["Ctrl","Shift","Alt","Meta"];static Keys={Backspace:"Backspace",Tab:"Tab",LeftControl:"ControlLeft",RightControl:"ControlRight",LeftShift:"ShiftLeft",RightShift:"ShiftRight",LeftAlt:"AltLeft",RightAlt:"AltRight",Enter:"Enter",Pause:"Pause",CapsLock:"CapsLock",Escape:"Escape",Space:"Space",PageUp:"PageUp",PageDown:"PageDown",End:"End",Home:"Home",ArrowLeft:"Left",ArrowUp:"Up",ArrowRight:"Right",ArrowDown:"Down",PrintScreen:"PrintScreen",Insert:"Insert",Delete:"Delete",Zero:"Digit0",One:"Digit1",Two:"Digit2",Three:"Digit3",Four:"Digit4",Five:"Digit5",Six:"Digit6",Seven:"Digit7",Eight:"Digit8",Nine:"Digit9",A:"KeyA",B:"KeyB",C:"KeyC",D:"KeyD",E:"KeyE",F:"KeyF",G:"KeyG",H:"KeyH",I:"KeyI",K:"KeyK",L:"KeyL",M:"KeyM",N:"KeyN",O:"KeyO",P:"KeyP",Q:"KeyQ",R:"KeyR",S:"KeyS",T:"KeyT",U:"KeyU",V:"KeyV",W:"KeyW",X:"KeyX",Y:"KeyY",Z:"KeyZ",NumPadZero:"Numpad0",NumPadOne:"Numpad1",NumPadTwo:"Numpad2",NumPadThree:"Numpad3",NumPadFour:"Numpad4",NumPadFive:"Numpad5",NumPadSix:"Numpad6",NumPadSeven:"Numpad7",NumPadEight:"Numpad8",NumPadNine:"Numpad9",Multiply:"NumpadMultiply",Add:"NumpadAdd",Subtract:"NumpadSubtract",Decimal:"NumpadDecimal",Divide:"NumpadDivide",F1:"F1",F2:"F2",F3:"F3",F4:"F4",F5:"F5",F6:"F6",F7:"F7",F8:"F8",F9:"F9",F10:"F10",F11:"F11",F12:"F12",NumLock:"NumLock",ScrollLock:"ScrollLock"}}class ie{#e;get target(){return this.#e}#t;get blueprint(){return this.#t}options;listenHandler=()=>this.listenEvents();unlistenHandler=()=>this.unlistenEvents();constructor(e,t,i={}){i.consumeEvent??=!1,i.listenOnFocus??=!1,i.unlistenOnTextEdit??=!1,this.#e=e,this.#t=t,this.options=i}setup(){this.options.listenOnFocus&&(this.blueprint.addEventListener(te.focusEventName.begin,this.listenHandler),this.blueprint.addEventListener(te.focusEventName.end,this.unlistenHandler)),this.options.unlistenOnTextEdit&&(this.blueprint.addEventListener(te.editTextEventName.begin,this.unlistenHandler),this.blueprint.addEventListener(te.editTextEventName.end,this.listenHandler)),this.blueprint.focused&&this.listenEvents()}cleanup(){this.unlistenEvents(),this.blueprint.removeEventListener(te.focusEventName.begin,this.listenHandler),this.blueprint.removeEventListener(te.focusEventName.end,this.unlistenHandler),this.blueprint.removeEventListener(te.editTextEventName.begin,this.unlistenHandler),this.blueprint.removeEventListener(te.editTextEventName.end,this.listenHandler)}listenEvents(){}unlistenEvents(){}}class ne{static#i=new Map;static registerSerializer(e,t){ne.#i.set(e,t)}static getSerializer(e){return ne.#i.get(e)}}class se{constructor(e){this.attributes=e}}class re{#n;get types(){return this.#n}constructor(...e){this.#n=e}getFirstType(){return this.#n[0]}}class ae{static emptyObj={};static booleanConverter={fromAttribute:(e,t)=>{},toAttribute:(e,t)=>!0===e?"true":!1===e?"false":""};static sigmoid(e,t=1.7){return 1/(1+e/(1-e)**-t)}static sigmoidPositive(e,t=3.7,i=1.1){return 1-Math.exp(-((e/i)**t))}static clamp(e,t=-1/0,i=1/0){return Math.min(Math.max(e,t),i)}static getScale(e){const t=e.blueprint?e.blueprint.getScale():getComputedStyle(e).getPropertyValue("--ueb-scale");return""!=t?parseFloat(t):1}static minDecimals(e,t=1,i=1e-8){const n=e*10**t;return Math.abs(n%1)>i?e.toString():e.toFixed(t)}static numberFromText(e=""){switch(e=e.toLowerCase()){case"zero":return 0;case"one":return 1;case"two":return 2;case"three":return 3;case"four":return 4;case"five":return 5;case"six":return 6;case"seven":return 7;case"eight":return 8;case"nine":return 9}}static roundDecimals(e,t=1){const i=10**t;return Math.round(e*i)/i}static approximatelyEqual(e,t,i=1e-8){return!(Math.abs(e-t)>i)}static convertLocation(e,t,i=!1){const n=i?1:1/ae.getScale(t),s=t.getBoundingClientRect();return[Math.round((e[0]-s.x)*n),Math.round((e[1]-s.y)*n)]}static isSerialized(e,t,i=ae.objectGet(e.constructor.attributes,t)){return i?.constructor===Object&&i.serialized}static objectGet(e,t,i){if(void 0!==e){if(!(t instanceof Array))throw new TypeError("UEBlueprint: Expected keys to be an array");return e instanceof se&&(e=e.attributes),0!=t.length&&t[0]in e&&void 0!==e[t[0]]?1==t.length?e[t[0]]:ae.objectGet(e[t[0]],t.slice(1),i):i}}static objectSet(e,t,i,n=!1,s=Object){if(!(t instanceof Array))throw new TypeError("Expected keys to be an array.");if(1==t.length){if(n||t[0]in e||void 0===e[t[0]])return e[t[0]]=i,!0}else if(t.length>0)return!n||e[t[0]]instanceof Object||(e[t[0]]=new s),ae.objectSet(e[t[0]],t.slice(1),i,n,s);return!1}static equals(e,t){return e?.equals&&t?.equals?e.equals(t):(e=ae.sanitize(e),t=ae.sanitize(t),e?.constructor===BigInt&&t?.constructor===Number?t=BigInt(t):e?.constructor===Number&&t?.constructor===BigInt&&(e=BigInt(e)),e===t||e instanceof Array&&t instanceof Array&&(e.length===t.length&&e.every(((e,i)=>ae.equals(e,t[i])))))}static getType(e){return null===e?null:e?.constructor===Object&&e?.type instanceof Function?e.type:e?.constructor}static isValueOfType(e,t,i=!1){return i&&null===e||e instanceof t||e?.constructor===t}static sanitize(e,t=e?.constructor){if(t instanceof re){let i=t.types.find((t=>ae.isValueOfType(e,t,!1)));i||(i=t.getFirstType()),t=i}return t&&!ae.isValueOfType(e,t,!0)&&(e=t===BigInt?BigInt(e):new t(e)),(e instanceof Boolean||e instanceof Number||e instanceof String||e instanceof BigInt)&&(e=e.valueOf()),e}static snapToGrid(e,t,i){return 1===i?[e,t]:[i*Math.round(e/i),i*Math.round(t/i)]}static mergeArrays(e=[],t=[]){let i=[];e=[...e],t=[...t];e:for(;;){for(let n=0;n","\n").replaceAll(/(\)/g,"")}static capitalFirstLetter(e){return 0===e.length?e:e.charAt(0).toUpperCase()+e.slice(1)}static formatStringName(e=""){return e.replace(/^\s*b/,"").replaceAll(/^K2(?:Node|node)?_|(?<=[a-z])(?=[A-Z0-9])|(?<=[A-Z])(?=[A-Z][a-z]|[0-9])|(?<=[014-9]|(?:2|3)(?!D(?:[^a-z]|$)))(?=[a-zA-Z])|\s*_+\s*|\s{2,}/g," ").split(" ").map((e=>ae.capitalFirstLetter(e))).join(" ").trim()}static getIdFromReference(e){return e.replace(/(?:.+\.)?([^\.]+)$/,"$1").replaceAll(/(?<=[a-z\d])(?=[A-Z])|(?<=[a-zA-Z])(?=\d)|(?<=[A-Z]{2})(?=[A-Z][a-z])/g,"-").toLowerCase()}static printLinearColor(e){return`${Math.round(255*e.R.valueOf())}, ${Math.round(255*e.G.valueOf())}, ${Math.round(255*e.B.valueOf())}`}static getPolarCoordinates(e,t,i=!1){let n=Math.atan2(t,e);return i&&n<0&&(n=2*Math.PI+n),[Math.sqrt(e*e+t*t),n]}static getCartesianCoordinates(e,t){return[e*Math.cos(t),e*Math.sin(t)]}static range(e=0,t=0,i=(t>=e?1:-1)){return Array.from({length:Math.ceil((t-e)/i)},((t,n)=>e+n*i))}static paste(e,t){const i=new ClipboardEvent("paste",{bubbles:!0,cancelable:!0,clipboardData:new DataTransfer});i.clipboardData.setData("text",t),e.dispatchEvent(i)}static animate(e,t,i,n,s=(e=>{const t=e**3.5;return t/(t+(1-e)**3.5)})){let r;const a=o=>{void 0===r&&(r=o);let l=(o-r)/i;ae.approximatelyEqual(l,1)||l>1?l=1:requestAnimationFrame(a);const u=e+(t-e)*s(l);n(u)};requestAnimationFrame(a)}}class oe{static attributes={};static defaultAttribute={showDefault:!0,nullable:!1,ignored:!1,serialized:!1,expected:!1};constructor(e={},t=!1){const i=(e,n,s={},r="")=>{const a=Object.keys(s),o=Object.keys(n),l=ae.mergeArrays(o,a);for(let o of l){let l=ae.objectGet(s,[o]),u=n[o];if(u instanceof se){e[o]={},i(e[o],u.attributes,s[o],o+".");continue}if(t||(o in n?a.length>0&&!(o in s)&&u.showDefault&&!u.ignored&&console.warn(`UEBlueprint: ${this.constructor.name} will add attribute ${r}${o} not defined in the serialized data`):console.warn(`UEBlueprint: Attribute ${r}${o} in the serialized data is not defined in ${this.constructor.name}.attributes`)),!u){e[o]=l;continue}let c=u.value,h=u.type;u.serialized&&h instanceof Function&&(h=h(this)),h instanceof Array&&(h=Array),c instanceof Function&&(c=c(this)),void 0===h&&(h=ae.getType(c));const d=u.predicate?t=>{Object.defineProperties(e,{["#"+o]:{writable:!0,enumerable:!1},[o]:{enumerable:!0,get(){return this["#"+o]},set(e){u.predicate?.(e)?this["#"+o]=e:console.warn(`UEBlueprint: Tried to assign attribute ${r}${o} to ${this.constructor.name} not satisfying the predicate`)}}}),this[o]=t}:t=>e[o]=t;void 0===l?(h instanceof re&&(h=null!=c?h.types.find((e=>c instanceof e||c.constructor==e))??h.getFirstType():h.getFirstType()),void 0===c&&(c=ae.sanitize(new h)),u.showDefault?(u.serialized&&h!==String&&c.constructor===String&&(c=ne.getSerializer(h).deserialize(c)),d(ae.sanitize(c,h))):d(void 0)):(l?.constructor===String&&u.serialized&&h!==String&&(l=ne.getSerializer(h).deserialize(l)),d(ae.sanitize(l,h)))}},n=this.constructor.attributes;e.constructor!==Object&&1===Object.keys(n).length&&(e={[Object.keys(n)[0]]:e}),i(this,n,e)}static cleanupAttributes(e,t=""){for(const i in e){if(e[i]instanceof se){this.cleanupAttributes(e[i].attributes,t+"."+i);continue}e[i].constructor!==Object&&(e[i]={value:e[i]});const n=e[i];if(void 0!==n.type||n.value instanceof Function||(n.type=ae.getType(n.value)),e[i]={...oe.defaultAttribute,...n},void 0===n.value&&void 0===n.type)throw new Error(`UEBlueprint: Expected either "type" or "value" property in ${this.name} attribute ${t}`+i);null===n.value&&(e[i].nullable=!0)}}static isValueOfType(e,t){return null!=e&&(e instanceof t||e.constructor===t)}static expectsAllKeys(){return!Object.values(this.attributes).filter((e=>!e.ignored)).some((e=>!e.expected))}unexpectedKeys(){return Object.keys(this).length-Object.keys(this.constructor.attributes).length}equals(e){const t=Object.keys(this),i=Object.keys(this);if(t.length!=i.length)return!1;for(const i of t){if(this[i]instanceof oe&&!this[i].equals(e[i]))return!1;if(!ae.equals(this[i],e[i]))return!1}return!0}}class le extends oe{static attributes={...super.attributes,value:{value:0,predicate:e=>e%1==0&&e>1<<31&&e<-(1<<31)}};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e),this.value}valueOf(){return this.value}toString(){return this.value.toString()}}class ue extends le{static attributes={...super.attributes,value:{...super.attributes.value,predicate:e=>e%1==0&&e>=0&&e<256}};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e)}}class ce extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class he extends ce{}class de extends oe{static lookbehind="INVTEXT";static attributes={value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}}class pe extends oe{static lookbehind="NSLOCTEXT";static attributes={namespace:"",key:"",value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.namespace,this.key,this.value}toString(){return ae.capitalFirstLetter(this.value)}}class me extends oe{static lookbehind="LOCGEN_FORMAT_NAMED";static attributes={value:{type:[new re(pe,de,me)]}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}}class ge extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}static generateGuid(e=!0){let t=new Uint32Array(4);!0===e&&crypto.getRandomValues(t);let i="";return t.forEach((e=>{i+=("0".repeat(8)+e.toString(16).toUpperCase()).slice(-8)})),new ge({value:i})}constructor(e){e||(e=ge.generateGuid().value),super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class be extends oe{static attributes={type:"",path:""};static{this.cleanupAttributes(this.attributes)}constructor(e={}){e.constructor===String&&(e={path:e}),super(e),this.type,this.path}getName(){return this.path.match(/[^\.\/]+$/)?.[0]??""}}class ve extends oe{static attributes={MemberParent:{type:be,showDefault:!1},MemberName:{type:String,showDefault:!1},MemberGuid:{type:ge,showDefault:!1}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.MemberParent,this.MemberName}}class fe extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}static attributeConverter={fromAttribute:(e,t)=>new fe(e),toAttribute:(e,t)=>e.toString()};constructor(e){super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class ye extends oe{static attributes={...super.attributes,value:{value:0n,predicate:e=>e>=-(1n<<63n)&&e<1n<<63n}};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e),this.value}valueOf(){return this.value}toString(){return this.value.toString()}}class we extends oe{static attributes={ActionName:"",bShift:!1,bCtrl:!1,bAlt:!1,bCmd:!1,Key:{type:fe}};static{this.cleanupAttributes(this.attributes)}constructor(e={}){e.ActionName=e.ActionName??"",e.bShift=e.bShift??!1,e.bCtrl=e.bCtrl??!1,e.bAlt=e.bAlt??!1,e.bCmd=e.bCmd??!1,super(e),this.ActionName,this.bShift,this.bCtrl,this.bAlt,this.bCmd,this.Key}}class Ee extends oe{static attributes={value:0};static{this.cleanupAttributes(this.attributes)}constructor(e=0){super(e),this.value=ae.clamp(this.value,0,1)}valueOf(){return this.value}toString(){return this.value.toFixed(6)}}class Ce extends oe{static attributes={R:{type:Ee,expected:!0},G:{type:Ee,expected:!0},B:{type:Ee,expected:!0},A:{type:Ee,value:()=>new Ee(1)},H:{type:Ee,showDefault:!0,ignored:!0},S:{type:Ee,showDefault:!0,ignored:!0},V:{type:Ee,showDefault:!0,ignored:!0}};static{this.cleanupAttributes(this.attributes)}static linearToSRGB(e){return e<=0?0:e>=1?1:e<.0031308?12.92*e:1.055*Math.pow(e,1/2.4)-.055}static sRGBtoLinear(e){return e<=0?0:e>=1?1:e<.04045?e/12.92:Math.pow((e+.055)/1.055,2.4)}static getWhite(){return new Ce({R:1,G:1,B:1})}constructor(e){e instanceof Array&&(e={R:e[0]??0,G:e[1]??0,B:e[2]??0,A:e[3]??1}),super(e),this.R,this.G,this.B,this.A,this.H,this.S,this.V,this.#s()}#s(){const e=this.R.value,t=this.G.value,i=this.B.value;if(ae.approximatelyEqual(e,t)&&ae.approximatelyEqual(e,i)&&ae.approximatelyEqual(t,i))return this.S.value=0,void(this.V.value=e);const n=Math.max(e,t,i),s=Math.min(e,t,i),r=n-s;let a;switch(n){case s:a=0;break;case e:a=(t-i)/r+(te.toString(16).toUpperCase().padStart(2,"0"))).join("")}toSRGBAString(){return this.toSRGBA().map((e=>e.toString(16).toUpperCase().padStart(2,"0"))).join("")}toHSVA(){return[this.H.value,this.S.value,this.V.value,this.A.value]}toNumber(){return(Math.round(255*this.R.value)<<24)+(Math.round(255*this.G.value)<<16)+(Math.round(255*this.B.value)<<8)+Math.round(255*this.A.value)}setFromRGBANumber(e){this.A.value=(255&e)/255,this.B.value=(e>>8&255)/255,this.G.value=(e>>16&255)/255,this.R.value=(e>>24&255)/255,this.#s()}setFromSRGBANumber(e){this.A.value=(255&e)/255,this.B.value=Ce.sRGBtoLinear((e>>8&255)/255),this.G.value=Ce.sRGBtoLinear((e>>16&255)/255),this.R.value=Ce.sRGBtoLinear((e>>24&255)/255),this.#s()}toString(){return ae.printLinearColor(this)}}class Se extends oe{static attributes={MacroGraph:{type:be},GraphBlueprint:{type:be},GraphGuid:{type:ge}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.MacroGraph,this.GraphBlueprint,this.GuidEntity}getMacroName(){const e=this.MacroGraph.path.search(":");return this.MacroGraph.path.substring(e+1)}}class xe extends oe{static attributes={value:""};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.value}valueOf(){return this.value}toString(){return this.value}}class Pe extends oe{static attributes={objectName:{type:xe},pinGuid:{type:ge}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.objectName,this.pinGuid}}class ke extends oe{static attributes={TerminalCategory:{value:"",showDefault:!1},TerminalSubCategory:{value:"",showDefault:!1},bTerminalIsConst:{value:!1,showDefault:!1},bTerminalIsWeakPointer:{value:!1,showDefault:!1},bTerminalIsUObjectWrapper:{value:!1,showDefault:!1}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.TerminalCategory,this.TerminalSubCategory,this.bTerminalIsConst,this.bTerminalIsWeakPointer,this.bTerminalIsUObjectWrapper}}class Le extends oe{static attributes={R:{value:0},P:{value:0},Y:{value:0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.R,this.P,this.Y}getRoll(){return this.R}getPitch(){return this.P}getYaw(){return this.Y}}class Ne extends Le{}class Te extends oe{static attributes={X:{value:0,expected:!0},Y:{value:0,expected:!0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.X,this.Y}}class Ae extends Te{}class Me extends oe{static attributes={X:{value:0,expected:!0},Y:{value:0,expected:!0},Z:{value:0,expected:!0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.X,this.Y,this.Z}}class Be extends Me{}class De extends oe{static#r={"/Script/CoreUObject.LinearColor":Ce,"/Script/CoreUObject.Rotator":Le,"/Script/CoreUObject.Vector":Me,"/Script/CoreUObject.Vector2D":Te,bool:Boolean,byte:ue,enum:he,exec:String,int:le,int64:ye,name:String,real:Number,string:String};static#a={"/Script/CoreUObject.Vector2D":Ae,"/Script/CoreUObject.Vector":Be,"/Script/CoreUObject.Rotator":Ne};static lookbehind="Pin";static attributes={PinId:{type:ge},PinName:"",PinFriendlyName:{type:new re(pe,me,String),showDefault:!1},PinToolTip:{type:String,showDefault:!1},Direction:{type:String,showDefault:!1},PinType:new se({PinCategory:"",PinSubCategory:"",PinSubCategoryObject:{type:be},PinSubCategoryMemberReference:{type:ve,value:null},PinValueType:{type:ke,value:null},ContainerType:{type:xe},bIsReference:!1,bIsConst:!1,bIsWeakPointer:!1,bIsUObjectWrapper:!1,bSerializeAsSinglePrecisionFloat:!1}),LinkedTo:{type:[Pe],showDefault:!1},DefaultValue:{type:e=>e.getEntityType(!0)??String,serialized:!0,showDefault:!1},AutogeneratedDefaultValue:{type:String,showDefault:!1},DefaultObject:{type:be,showDefault:!1,value:null},PersistentGuid:{type:ge},bHidden:!1,bNotConnectable:!1,bDefaultValueIsReadOnly:!1,bDefaultValueIsIgnored:!1,bAdvancedView:!1,bOrphanedPin:!1};static{this.cleanupAttributes(this.attributes)}constructor(e={},t=!1){super(e,t),this.PinId,this.PinName,this.PinFriendlyName,this.PinToolTip,this.Direction,this.PinType,this.LinkedTo,this.DefaultValue,this.AutogeneratedDefaultValue,this.DefaultObject,this.PersistentGuid,this.bHidden,this.bNotConnectable,this.bDefaultValueIsReadOnly,this.bDefaultValueIsIgnored,this.bAdvancedView,this.bOrphanedPin}getType(){const e=this.PinType.PinSubCategoryObject;return"struct"===this.PinType.PinCategory||"object"===this.PinType.PinCategory?e.path:"byte"!==this.PinType.PinCategory||e.type!==te.nodeType.enum&&e.type!==te.nodeType.userDefinedEnum?this.PinType.PinCategory:"enum"}getEntityType(e=!1){const t=this.getType(),i=De.#r[t],n=De.#a[t];return e&&void 0!==n?n:i}getDisplayName(){let e=null;return this.PinToolTip&&(e=this.PinToolTip.match(/\s*(.+?(?=\n)|.+\S)\s*/))?ae.formatStringName(e[1]):ae.formatStringName(this.PinName)}copyTypeFrom(e){this.PinType.PinCategory=e.PinType.PinCategory,this.PinType.PinSubCategory=e.PinType.PinSubCategory,this.PinType.PinSubCategoryObject=e.PinType.PinSubCategoryObject,this.PinType.PinSubCategoryMemberReference=e.PinType.PinSubCategoryMemberReference,this.PinType.PinValueType=e.PinType.PinValueType,this.PinType.ContainerType=e.PinType.ContainerType,this.PinType.bIsReference=e.PinType.bIsReference,this.PinType.bIsConst=e.PinType.bIsConst,this.PinType.bIsWeakPointer=e.PinType.bIsWeakPointer,this.PinType.bIsUObjectWrapper=e.PinType.bIsUObjectWrapper,this.PinType.bSerializeAsSinglePrecisionFloat=e.PinType.bSerializeAsSinglePrecisionFloat}getDefaultValue(e=!1){return void 0===this.DefaultValue&&e&&(this.DefaultValue=new(this.getEntityType(!0))),this.DefaultValue}isExecution(){return"exec"===this.PinType.PinCategory}isHidden(){return this.bHidden}isInput(){return!this.bHidden&&"EGPD_Output"!=this.Direction}isOutput(){return!this.bHidden&&"EGPD_Output"==this.Direction}isLinked(){return this.LinkedTo?.length>0??!1}linkTo(e,t){const i=this.LinkedTo?.some((i=>i.objectName.toString()==e&&i.pinGuid.valueOf()==t.PinId.valueOf()));return!i&&((this.LinkedTo??=[]).push(new Pe({objectName:e,pinGuid:t.PinId})),!0)}unlinkFrom(e,t){const i=this.LinkedTo?.findIndex((i=>i.objectName.toString()==e&&i.pinGuid.valueOf()==t.PinId.valueOf()));return i>=0&&(this.LinkedTo.splice(i,1),0!==this.LinkedTo.length||De.attributes.LinkedTo.showDefault||(this.LinkedTo=void 0),!0)}getSubCategory(){return this.PinType.PinSubCategoryObject.path}pinColor(){return te.pinColor[this.getType()]??te.pinColor[this.PinType.PinCategory]??te.pinColor.default}}class $e{static array=$``;static branchNode=$``;static breakStruct=$``;static cast=$``;static close=$``;static correct=$``;static delegate=$``;static doN=$``;static doOnce=$``;static enum=$``;static event=$``;static execPin=$``;static expandIcon=$``;static forEachLoop=$``;static functionSymbol=$``;static gamepad=$``;static genericPin=$``;static keyboard=$``;static loop=$``;static macro=$``;static map=$``;static makeArray=$``;static makeMap=$``;static makeSet=$``;static makeStruct=$``;static mouse=$``;static questionMark=$``;static referencePin=$``;static reject=$``;static set=$``;static select=$``;static sequence=$``;static touchpad=$``}class He extends oe{static attributes={MemberScope:{value:"",showDefault:!1},MemberName:"",MemberGuid:{type:ge},bSelfContext:{value:!1,showDefault:!1}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.MemberName,this.GuidEntity,this.bSelfContext}}class Oe extends oe{static attributes={Class:{type:be},Name:"",AxisKey:{type:ce,showDefault:!1},InputAxisKey:{type:ce,showDefault:!1},bIsPureFunc:{value:!1,showDefault:!1},bIsConstFunc:{value:!1,showDefault:!1},VariableReference:{type:He,value:null,showDefault:!1},SelfContextInfo:{type:ce,value:null,showDefault:!1},DelegatePropertyName:{type:String,showDefault:!1},DelegateOwnerClass:{type:be,showDefault:!1},ComponentPropertyName:{type:String,showDefault:!1},EventReference:{type:ve,value:null,showDefault:!1},FunctionReference:{type:ve,value:null,showDefault:!1},CustomFunctionName:{type:String,showDefault:!1},TargetType:{type:be,value:null,showDefault:!1},MacroGraphReference:{type:Se,value:null,showDefault:!1},Enum:{type:be,showDefault:!1},InputKey:{type:ce,showDefault:!1},bOverrideFunction:{type:Boolean,showDefault:!1},bInternalEvent:{type:Boolean,showDefault:!1},bConsumeInput:{type:Boolean,showDefault:!1},bExecuteWhenPaused:{type:Boolean,showDefault:!1},bOverrideParentBinding:{type:Boolean,showDefault:!1},bControl:{type:Boolean,showDefault:!1},bAlt:{type:Boolean,showDefault:!1},bShift:{type:Boolean,showDefault:!1},bCommand:{type:Boolean,showDefault:!1},CommentColor:{type:Ce,showDefault:!1},bCommentBubbleVisible_InDetailsPanel:{type:Boolean,showDefault:!1},bColorCommentBubble:{type:Boolean,value:!1,showDefault:!1},MoveMode:{type:ce,showDefault:!1},NodePosX:{type:le,showDefault:!1},NodePosY:{type:le,showDefault:!1},NodeWidth:{type:le,showDefault:!1},NodeHeight:{type:le,showDefault:!1},bCommentBubblePinned:{type:Boolean,showDefault:!1},bCommentBubbleVisible:{type:Boolean,showDefault:!1},NodeComment:{type:String,showDefault:!1},AdvancedPinDisplay:{type:fe,value:null,showDefault:!1},EnabledState:{type:fe,value:null,showDefault:!1},NodeGuid:{type:ge},ErrorType:{type:le,showDefault:!1},ErrorMsg:{type:String,value:"",showDefault:!1},CustomProperties:{type:[De]}};static nameRegex=/^(\w+?)(?:_(\d+))?$/;static sequencerScriptingNameRegex=/\/Script\/SequencerScripting\.MovieSceneScripting(.+)Channel/;static#o={A_AccentGrave:"à",Add:"Num +",C_Cedille:"ç",Decimal:"Num .",Divide:"Num /",E_AccentAigu:"é",E_AccentGrave:"è",F1:"F1",F10:"F10",F11:"F11",F12:"F12",F2:"F2",F3:"F3",F4:"F4",F5:"F5",F6:"F6",F7:"F7",F8:"F8",F9:"F9",Gamepad_Special_Left_X:"Touchpad Button X Axis",Gamepad_Special_Left_Y:"Touchpad Button Y Axis",Mouse2D:"Mouse XY 2D-Axis",Multiply:"Num *",Section:"§",Subtract:"Num -",Tilde:"`"};static{this.cleanupAttributes(this.attributes)}static keyName(e){let t=Oe.#o[e];if(t)return t;if(t=ae.numberFromText(e)?.toString(),t)return t;const i=e.match(/NumPad([a-zA-Z]+)/);return i&&(t=ae.numberFromText(i[1]),t)?"Num "+t:void 0}constructor(e,t=!1){super(e,t),this.Class,this.Name,this.AxisKey,this.InputAxisKey,this.bIsPureFunc,this.bIsConstFunc,this.VariableReference,this.SelfContextInfo,this.DelegatePropertyName,this.DelegateOwnerClass,this.ComponentPropertyName,this.EventReference,this.FunctionReference,this.CustomFunctionName,this.TargetType,this.MacroGraphReference,this.Enum,this.InputKey,this.bOverrideFunction,this.bInternalEvent,this.bConsumeInput,this.bExecuteWhenPaused,this.bOverrideParentBinding,this.bControl,this.bAlt,this.bShift,this.bCommand,this.CommentColor,this.bCommentBubbleVisible_InDetailsPanel,this.NodePosX,this.NodePosY,this.NodeWidth,this.NodeHeight,this.bCommentBubblePinned,this.bCommentBubbleVisible,this.NodeComment,this.AdvancedPinDisplay,this.EnabledState,this.NodeGuid,this.ErrorType,this.ErrorMsg,this.CustomProperties}getClass(){return this.Class.path}getType(){let e=this.getClass();return this.MacroGraphReference?.MacroGraph?.path?this.MacroGraphReference.MacroGraph.path:e}getObjectName(e=!1){return e?this.getNameAndCounter()[0]:this.Name}getNameAndCounter(){const e=this.getObjectName(!1).match(Oe.nameRegex);let t="",i=null;return e?(e.length>1&&(t=e[1]),e.length>2&&(i=parseInt(e[2])),[t,i]):["",0]}getCounter(){return this.getNameAndCounter()[1]}getNodeWidth(){return this.NodeWidth??this.getType()==te.nodeType.comment?te.defaultCommentWidth:void 0}setNodeWidth(e){this.NodeWidth||(this.NodeWidth=new le),this.NodeWidth.value=e}getNodeHeight(){return this.NodeHeight??this.getType()==te.nodeType.comment?te.defaultCommentHeight:void 0}setNodeHeight(e){this.NodeHeight||(this.NodeHeight=new le),this.NodeHeight.value=e}getNodePosX(){return this.NodePosX?.value??0}setNodePosX(e){this.NodePosX||(this.NodePosX=new le),this.NodePosX.value=Math.round(e)}getNodePosY(){return this.NodePosY?.value??0}setNodePosY(e){this.NodePosY||(this.NodePosY=new le),this.NodePosY.value=Math.round(e)}isEvent(){return this.getClass()===te.nodeType.event||this.getClass()===te.nodeType.customEvent||!!this.getDelegatePin()}isDevelopmentOnly(){const e=this.getClass();return"DevelopmentOnly"===this.EnabledState?.toString()||e.includes("Debug",Math.max(0,e.lastIndexOf(".")))}getHIDAttribute(){return this.InputKey??this.AxisKey??this.InputAxisKey}getDelegatePin(){return this.CustomProperties?.find((e=>"delegate"===e.PinType.PinCategory))}nodeDisplayName(){switch(this.getType()){case te.nodeType.componentBoundEvent:return`${ae.formatStringName(this.DelegatePropertyName)} (${this.ComponentPropertyName})`;case te.nodeType.dynamicCast:return this.TargetType?`Cast To ${this.TargetType?.getName()}`:"Bad cast node";case te.nodeType.enumLiteral:return`Literal enum ${this.Enum?.getName()}`;case te.nodeType.event:return`Event ${(this.EventReference?.MemberName??"").replace(/^Receive/,"")}`;case te.nodeType.executionSequence:return"Sequence";case te.nodeType.forEachElementInEnum:return`For Each ${this.Enum?.getName()}`;case te.nodeType.forEachLoopWithBreak:return"For Each Loop with Break";case te.nodeType.ifThenElse:return"Branch";case te.nodeType.variableGet:return"";case te.nodeType.variableSet:return"SET"}const e=this.getHIDAttribute();if(e){const t=e.toString();let i=Oe.keyName(t)??ae.formatStringName(t);return this.getClass()===te.nodeType.inputDebugKey?i="Debug Key "+i:this.getClass()===te.nodeType.getInputAxisKeyValue&&(i="Get "+i),i}if(this.getClass()===te.nodeType.macro)return ae.formatStringName(this.MacroGraphReference?.getMacroName());let t=this.FunctionReference?.MemberName;if(t){const e=this.FunctionReference.MemberParent?.path??"";switch(t){case"AddKey":{let t=e.match(Oe.sequencerScriptingNameRegex);if(t)return`Add Key (${ae.formatStringName(t[1])})`}break;case"LineTraceSingle":return"Line Trace By Channel";case"LineTraceSingleByProfile":return"Line Trace By Profile"}switch(e){case"/Script/Engine.KismetMathLibrary":if(t.startsWith("Conv_"))return"";if(t.startsWith("Percent_"))return"%";if(t.startsWith("EqualEqual_"))return"==";const e=t.match(/[BF]([A-Z]\w+)/);switch(e&&(t=e[1]),t){case"Abs":return"ABS";case"Exp":return"e";case"LineTraceSingle":return"Line Trace By Channel";case"Max":case"MaxInt64":return"MAX";case"Min":case"MinInt64":return"MIN"}break;case"/Script/Engine.BlueprintSetLibrary":{const e=t.match(/Set_(\w+)/);if(e)return ae.formatStringName(e[1]).toUpperCase()}break;case"/Script/Engine.BlueprintMapLibrary":{const e=t.match(/Map_(\w+)/);if(e)return ae.formatStringName(e[1]).toUpperCase()}}return ae.formatStringName(t)}return ae.formatStringName(this.getNameAndCounter()[0])}nodeColor(){switch(this.getClass()){case te.nodeType.callFunction:return this.bIsPureFunc?te.nodeColors.green:te.nodeColors.blue;case te.nodeType.event:case te.nodeType.customEvent:case te.nodeType.inputKey:case te.nodeType.inputAxisKeyEvent:case te.nodeType.inputDebugKey:return te.nodeColors.red;case te.nodeType.enumLiteral:case te.nodeType.makeArray:case te.nodeType.makeMap:case te.nodeType.select:return te.nodeColors.green;case te.nodeType.executionSequence:case te.nodeType.ifThenElse:case te.nodeType.macro:case te.nodeType.multiGate:return te.nodeColors.gray;case te.nodeType.dynamicCast:return te.nodeColors.turquoise}return this.bIsPureFunc?te.nodeColors.green:this.isEvent()?te.nodeColors.red:te.nodeColors.blue}nodeIcon(){switch(this.getType()){case te.nodeType.customEvent:return $e.event;case te.nodeType.doN:return $e.doN;case te.nodeType.doOnce:return $e.doOnce;case te.nodeType.dynamicCast:return $e.cast;case te.nodeType.enumLiteral:return $e.enum;case te.nodeType.event:return $e.event;case te.nodeType.executionSequence:case te.nodeType.multiGate:return $e.sequence;case te.nodeType.forEachElementInEnum:case te.nodeType.forLoop:case te.nodeType.forLoopWithBreak:case te.nodeType.whileLoop:return $e.loop;case te.nodeType.forEachLoop:case te.nodeType.forEachLoopWithBreak:return $e.forEachLoop;case te.nodeType.ifThenElse:return $e.branchNode;case te.nodeType.isValid:return $e.questionMark;case te.nodeType.makeArray:return $e.makeArray;case te.nodeType.makeMap:return $e.makeMap;case te.nodeType.makeSet:return $e.makeSet;case te.nodeType.select:return $e.select}if(this.nodeDisplayName().startsWith("Break"))return $e.breakStruct;if(this.getClass()===te.nodeType.macro)return $e.macro;const e=this.getHIDAttribute()?.toString();return e?e.includes("Mouse")?$e.mouse:e.includes("Gamepad_Special")?$e.keyboard:e.includes("Gamepad")||e.includes("Steam")?$e.gamepad:e.includes("Touch")?$e.touchpad:$e.keyboard:this.getDelegatePin()?$e.event:$e.functionSymbol}}"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self&&self;function Ve(e){return e&&e.__esModule&&Object.prototype.hasOwnProperty.call(e,"default")?e.default:e}var ze={exports:{}};"undefined"!=typeof self&&self;var Ie=Ve(ze.exports=function(e){var t={};function i(n){if(t[n])return t[n].exports;var s=t[n]={i:n,l:!1,exports:{}};return e[n].call(s.exports,s,s.exports,i),s.l=!0,s.exports}return i.m=e,i.c=t,i.d=function(e,t,n){i.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:n})},i.r=function(e){Object.defineProperty(e,"__esModule",{value:!0})},i.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return i.d(t,"a",t),t},i.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},i.p="",i(i.s=0)}([function(e,t,i){function n(e){if(!(this instanceof n))return new n(e);this._=e}var s=n.prototype;function r(e,t){for(var i=0;i>7),buf:function(e){var t=a((function(e,t,i,n){return e.concat(i===n.length-1?Buffer.from([t,0]).readUInt16BE(0):n.readUInt16BE(i))}),[],e);return Buffer.from(o((function(e){return(e<<1&65535)>>8}),t))}(i.buf)}})),i}function u(){return"undefined"!=typeof Buffer}function c(){if(!u())throw new Error("Buffer global does not exist; please use webpack if you need to parse Buffers in the browser.")}function h(e){c();var t=a((function(e,t){return e+t}),0,e);if(t%8!=0)throw new Error("The bits ["+e.join(", ")+"] add up to "+t+" which is not an even number of bytes; the total should be divisible by 8");var i,s=t/8,r=(i=function(e){return e>48},a((function(e,t){return e||(i(t)?t:e)}),null,e));if(r)throw new Error(r+" bit range requested exceeds 48 bit (6 byte) Number max.");return new n((function(t,i){var n=s+i;return n>t.length?C(i,s.toString()+" bytes"):E(n,a((function(e,t){var i=l(t,e.buf);return{coll:e.coll.concat(i.v),buf:i.buf}}),{coll:[],buf:t.slice(i,n)},e).coll)}))}function d(e,t){return new n((function(i,n){return c(),n+t>i.length?C(n,t+" bytes for "+e):E(n+t,i.slice(n,n+t))}))}function p(e,t){if("number"!=typeof(i=t)||Math.floor(i)!==i||t<0||t>6)throw new Error(e+" requires integer length in range [0, 6].");var i}function m(e){return p("uintBE",e),d("uintBE("+e+")",e).map((function(t){return t.readUIntBE(0,e)}))}function g(e){return p("uintLE",e),d("uintLE("+e+")",e).map((function(t){return t.readUIntLE(0,e)}))}function b(e){return p("intBE",e),d("intBE("+e+")",e).map((function(t){return t.readIntBE(0,e)}))}function v(e){return p("intLE",e),d("intLE("+e+")",e).map((function(t){return t.readIntLE(0,e)}))}function f(e){return e instanceof n}function y(e){return"[object Array]"==={}.toString.call(e)}function w(e){return u()&&Buffer.isBuffer(e)}function E(e,t){return{status:!0,index:e,value:t,furthest:-1,expected:[]}}function C(e,t){return y(t)||(t=[t]),{status:!1,index:-1,value:null,furthest:e,expected:t}}function S(e,t){if(!t)return e;if(e.furthest>t.furthest)return e;var i=e.furthest===t.furthest?function(e,t){if(function(){if(void 0!==n._supportsSet)return n._supportsSet;var e="undefined"!=typeof Set;return n._supportsSet=e,e}()&&Array.from){for(var i=new Set(e),s=0;s=0;){if(a in i){n=i[a].line,0===r&&(r=i[a].lineStart);break}("\n"===e.charAt(a)||"\r"===e.charAt(a)&&"\n"!==e.charAt(a+1))&&(s++,0===r&&(r=a+1)),a--}var o=n+s,l=t-r;return i[t]={line:o,lineStart:r},{offset:t,line:o+1,column:l+1}}function k(e){if(!f(e))throw new Error("not a parser: "+e)}function L(e,t){return"string"==typeof e?e.charAt(t):e[t]}function N(e){if("number"!=typeof e)throw new Error("not a number: "+e)}function T(e){if("function"!=typeof e)throw new Error("not a function: "+e)}function A(e){if("string"!=typeof e)throw new Error("not a string: "+e)}var M=2,B=3,D=8,$=5*D,H=4*D,O=" ";function V(e,t){return new Array(t+1).join(e)}function z(e,t,i){var n=t-e.length;return n<=0?e:V(i,n)+e}function I(e,t,i,n){return{from:e-t>0?e-t:0,to:e+i>n?n:e+i}}function R(e,t){var i,n,s,r,l,u=t.index,c=u.offset,h=1;if(c===e.length)return"Got the end of the input";if(w(e)){var d=c-c%D,p=c-d,m=I(d,$,H+D,e.length),g=o((function(e){return o((function(e){return z(e.toString(16),2,"0")}),e)}),function(e,t){var i=e.length,n=[],s=0;if(i<=t)return[e.slice()];for(var r=0;r=4&&(i+=1),h=2,s=o((function(e){return e.length<=4?e.join(" "):e.slice(0,4).join(" ")+" "+e.slice(4).join(" ")}),g),(l=(8*(r.to>0?r.to-1:r.to)).toString(16).length)<2&&(l=2)}else{var b=e.split(/\r\n|[\n\r\u2028\u2029]/);i=u.column-1,n=u.line-1,r=I(n,M,B,b.length),s=b.slice(r.from,r.to),l=r.to.toString().length}var v=n-r.from;return w(e)&&(l=(8*(r.to>0?r.to-1:r.to)).toString(16).length)<2&&(l=2),a((function(t,n,s){var a,o=s===v,u=o?"> ":O;return a=w(e)?z((8*(r.from+s)).toString(16),l,"0"):z((r.from+s+1).toString(),l," "),[].concat(t,[u+a+" | "+n],o?[O+V(" ",l)+" | "+z("",i," ")+V("^",h)]:[])}),[],s).join("\n")}function _(e,t){return["\n","-- PARSING FAILED "+V("-",50),"\n\n",R(e,t),"\n\n",(i=t.expected,1===i.length?"Expected:\n\n"+i[0]:"Expected one of the following: \n\n"+i.join(", ")),"\n"].join("");var i}function F(e){return void 0!==e.flags?e.flags:[e.global?"g":"",e.ignoreCase?"i":"",e.multiline?"m":"",e.unicode?"u":"",e.sticky?"y":""].join("")}function G(){for(var e=[].slice.call(arguments),t=e.length,i=0;i=2?N(t):t=0;var i=function(e){return RegExp("^(?:"+e.source+")",F(e))}(e),s=""+e;return n((function(e,n){var r=i.exec(e.slice(n));if(r){if(0<=t&&t<=r.length){var a=r[0],o=r[t];return E(n+a.length,o)}return C(n,"valid match group (0 to "+r.length+") in "+s)}return C(n,s)}))}function Y(e){return n((function(t,i){return E(i,e)}))}function q(e){return n((function(t,i){return C(i,e)}))}function J(e){if(f(e))return n((function(t,i){var n=e._(t,i);return n.index=i,n.value="",n}));if("string"==typeof e)return J(Z(e));if(e instanceof RegExp)return J(X(e));throw new Error("not a string, regexp, or parser: "+e)}function Q(e){return k(e),n((function(t,i){var n=e._(t,i),s=t.slice(i,n.index);return n.status?C(i,'not "'+s+'"'):E(i,null)}))}function ee(e){return T(e),n((function(t,i){var n=L(t,i);return i=e.length?C(t,"any character/byte"):E(t+1,L(e,t))})),re=n((function(e,t){return E(e.length,e.slice(t))})),ae=n((function(e,t){return t=0})).desc(t)},n.optWhitespace=he,n.Parser=n,n.range=function(e,t){return ee((function(i){return e<=i&&i<=t})).desc(e+"-"+t)},n.regex=X,n.regexp=X,n.sepBy=K,n.sepBy1=W,n.seq=G,n.seqMap=j,n.seqObj=function(){for(var e,t={},i=0,s=(e=arguments,Array.prototype.slice.call(e)),r=s.length,a=0;a255)throw new Error("Value specified to byte constructor ("+e+"=0x"+e.toString(16)+") is larger in value than a single byte.");var t=(e>15?"0x":"0x0")+e.toString(16);return n((function(i,n){var s=L(i,n);return s===e?E(n+1,s):C(n,t)}))},buffer:function(e){return d("buffer",e).map((function(e){return Buffer.from(e)}))},encodedString:function(e,t){return d("string",t).map((function(t){return t.toString(e)}))},uintBE:m,uint8BE:m(1),uint16BE:m(2),uint32BE:m(4),uintLE:g,uint8LE:g(1),uint16LE:g(2),uint32LE:g(4),intBE:b,int8BE:b(1),int16BE:b(2),int32BE:b(4),intLE:v,int8LE:v(1),int16LE:v(2),int32LE:v(4),floatBE:d("floatBE",4).map((function(e){return e.readFloatBE(0)})),floatLE:d("floatLE",4).map((function(e){return e.readFloatLE(0)})),doubleBE:d("doubleBE",8).map((function(e){return e.readDoubleBE(0)})),doubleLE:d("doubleLE",8).map((function(e){return e.readDoubleLE(0)}))},e.exports=n}]));class Re extends oe{static attributes={lookbehind:{value:"",showDefault:!1,ignore:!0}};static{this.cleanupAttributes(this.attributes)}constructor(e){super(e),this.lookbehind}}let _e=Ie;class Fe{static Regex={ByteInteger:/0*(?:25[0-5]|2[0-4]\d|1?\d?\d|)(?!\d)/,HexDigit:/[0-9a-fA-F]/,InlineOptWhitespace:/[^\S\n]*/,InlineWhitespace:/[^\S\n]+/,InsideString:/(?:[^"\\]|\\.)*/,Integer:/[\-\+]?\d+/,MultilineWhitespace:/\s*\n\s*/,Number:/[-\+]?\d+(?:\.\d+)?/,RealUnit:/^\+?(?:0(?:\.\d+)?|1(?:\.0+)?)(?![\.\d])/,Symbol:/[a-zA-Z_]\w*/,Word:/[a-zA-Z_]+/};static regexMap(e,t){const i=RegExp("^(?:"+e.source+")",e.flags),n=""+e;return _e(((e,s)=>{const r=i.exec(e.slice(s));return r?_e.makeSuccess(s+r[0].length,t(r)):_e.makeFailure(s,n)}))}static getStringParser(e){return _e(((t,i)=>{var n=i+e.length,s=t.slice(i,n);return s===e?makeSuccess(n,s):makeFailure(i,expected)}))}static getGrammarForType(e,t,i=e.AttributeAnyValue){switch(t){case BigInt:return e.BigInt;case Boolean:return e.Boolean;case ue:return e.Byte;case he:return e.Enum;case me:return e.FormatText;case ve:return e.FunctionReference;case ge:return e.Guid;case fe:return e.Identifier;case ye:return e.Integer64;case le:return e.Integer;case de:return e.InvariantText;case we:return e.KeyBinding;case Ce:return e.LinearColor;case pe:return e.LocalizedText;case Se:return e.MacroGraphReference;case Number:return e.Number;case be:return e.ObjectReference;case xe:return e.PathSymbol;case De:return e.Pin;case Pe:return e.PinReference;case ke:return e.PinType;case Ee:return e.RealUnit;case Le:return e.Rotator;case Ne:return e.SimpleSerializationRotator;case Ae:return e.SimpleSerializationVector2D;case Be:return e.SimpleSerializationVector;case String:return e.String;case ce:return e.Symbol;case He:return e.VariableReference;case Te:return e.Vector2D;case Me:return e.Vector}let n=i;const s=t.constructor===Object?t.type:t.constructor;return s instanceof Array?n=Fe.getGrammarForType(e,s[0]).trim(_e.optWhitespace).sepBy(_e.string(",")).skip(_e.regex(/,?\s*/)).wrap(_e.string("("),_e.string(")")):s instanceof re?n=s.types.map((t=>Fe.getGrammarForType(e,t))).reduce(((t,i)=>i&&t!==e.AttributeAnyValue?t.or(i):e.AttributeAnyValue)):t.constructor===Object&&(n=Fe.getGrammarForType(e,s,i)),t.constructor===Object&&(t.serialized&&s.constructor!==String&&(n=n.wrap(_e.string('"'),_e.string('"'))),t.nullable&&(n=n.or(e.Null))),n}static ReferencePath=(e,t)=>_e.alt(t,_e.seq(_e.string("/"),t.map((e=>e.toString())).sepBy1(_e.string(".")).tieWith(".").sepBy1(_e.string(":")).tieWith(":")).tie().atLeast(2).tie());static createAttributeGrammar=(e,t,i=_e.string("=").trim(_e.optWhitespace))=>e.AttributeName.skip(i).chain((i=>{const n=i.split("."),s=ae.objectGet(t.attributes,n);return(s?s.constructor===Object&&s.serialized?e.String:Fe.getGrammarForType(e,s,e.AttributeAnyValue):e.AttributeAnyValue).map((e=>t=>ae.objectSet(t,n,e,!0)))}));static createEntityGrammar=(e,t,i=!0)=>_e.seqMap(t.lookbehind?_e.seq(_e.string(t.lookbehind),_e.optWhitespace,_e.string("(")):_e.string("("),Fe.createAttributeGrammar(e,t).trim(_e.optWhitespace).sepBy(_e.string(",")).skip(_e.regex(/,?/).then(_e.optWhitespace)),_e.string(")"),((e,t,i)=>{let n={};return t.forEach((e=>e(n))),n})).chain((e=>{let n=Object.keys(e);if(Object.keys(t.attributes).filter((e=>t.attributes[e].expected)).find((e=>!n.includes(e))))return _e.fail();const s=Object.keys(e).filter((e=>!(e in t.attributes))).length;return!i&&s>0||i&&s+.5>Math.sqrt(n)?_e.fail():_e.succeed().map((()=>new t(e)))}));Null=e=>_e.regex(new RegExp(String.raw`\(${Fe.Regex.InlineOptWhitespace.source}\)`)).map((()=>null)).desc("null: ()");Boolean=e=>Fe.regexMap(/(true)|false/i,(e=>!!e[1])).desc("either True or False");Number=e=>_e.regex(Fe.Regex.Number).map(Number).desc("a number");BigInt=e=>_e.regex(Fe.Regex.Integer).map((e=>BigInt(e))).desc("a big integer");RealUnit=e=>_e.regex(Fe.Regex.RealUnit).map(Number).desc("a number between 0 and 1");NaturalNumber=e=>Fe.regexMap(/\d+/,(e=>parseInt(e[0]))).desc("a natural number");ColorNumber=e=>_e.regexp(Fe.Regex.ByteInteger).desc("a number between 0 and 255");Word=e=>_e.regexp(Fe.Regex.Word).desc("a word");String=e=>_e.regexp(new RegExp(`"(${Fe.Regex.InsideString.source})"`),1).map(ae.unescapeString).desc('string (with possibility to escape the quote using ")');AttributeName=e=>_e.regexp(new RegExp(String.raw`(?:(?:^|(?_e.string("None").map((()=>new be({type:"None",path:""}))).desc("none");Integer64=e=>e.BigInt.map((e=>new ye(e))).desc("an integer64");Integer=e=>_e.regex(Fe.Regex.Integer).map((e=>new le(e))).desc("an integer");Byte=e=>_e.regex(Fe.Regex.ByteInteger).map((e=>new ue(parseInt(e)))).desc("a Byte");Guid=e=>_e.regexp(new RegExp(`${Fe.Regex.HexDigit.source}{32}`)).map((e=>new ge({value:e}))).desc("32 digit hexadecimal value");Identifier=e=>_e.regex(/\w+/).map((e=>new fe(e)));PathSymbol=e=>_e.regex(/\w+/).map((e=>new xe({value:e})));PathSymbolOptSpaces=e=>_e.regex(/(?:(?:^|(?new xe({value:e})));Symbol=e=>_e.regex(Fe.Regex.Symbol).map((e=>new ce({value:e})));Enum=e=>_e.regex(Fe.Regex.Symbol).map((e=>new he({value:e})));ObjectReference=e=>_e.alt(e.None,...[Fe.ReferencePath(e,e.PathSymbolOptSpaces).map((e=>new be({type:"",path:e})))].flatMap((e=>[e.wrap(_e.string('"'),_e.string('"')),e.wrap(_e.string("'\""),_e.string("\"'"))])),_e.seqMap(Fe.ReferencePath(e,e.PathSymbolOptSpaces),_e.optWhitespace,_e.alt(...[Fe.ReferencePath(e,e.PathSymbolOptSpaces)].flatMap((e=>[e.wrap(_e.string('"'),_e.string('"')),e.wrap(_e.string("'\""),_e.string("\"'"))]))),((e,t,i)=>new be({type:e,path:i}))),Fe.ReferencePath(e,e.PathSymbol).map((e=>new be({type:"",path:e}))),e.Word.map((e=>new be({type:e,path:""}))));LocalizedText=e=>Fe.regexMap(new RegExp(String.raw`${pe.lookbehind}\s*\(`+String.raw`\s*"(${Fe.Regex.InsideString.source})"\s*,`+String.raw`\s*"(${Fe.Regex.InsideString.source})"\s*,`+String.raw`\s*"(${Fe.Regex.InsideString.source})"\s*\)`),(e=>new pe({namespace:e[1],key:e[2],value:e[3]})));InvariantText=e=>Fe.regexMap(new RegExp(String.raw`${de.lookbehind}\s*\("(${Fe.Regex.InsideString.source})"\)`),(e=>new de({value:e[1]})));FormatText=e=>_e.string(me.lookbehind).then(_e.optWhitespace).then(Fe.getGrammarForType(e,me.attributes.value).wrap(_e.string("("),_e.string(")")));AttributeAnyValue=e=>_e.alt(e.Boolean,e.Guid,e.None,e.Null,e.Number,e.String,e.LocalizedText,e.InvariantText,e.PinReference,e.Vector,e.LinearColor,e.Vector2D,e.UnknownKeys,e.ObjectReference,e.Symbol);PinReference=e=>_e.seqMap(e.PathSymbol,_e.whitespace,e.Guid,((e,t,i)=>new Pe({objectName:e,pinGuid:i})));PinType=e=>Fe.createEntityGrammar(e,ke,!0);Vector2D=e=>Fe.createEntityGrammar(e,Te,!1);Vector=e=>Fe.createEntityGrammar(e,Me,!1);Rotator=e=>Fe.createEntityGrammar(e,Le,!1);SimpleSerializationRotator=e=>_e.seqMap(e.Number,_e.string(",").trim(_e.optWhitespace),e.Number,_e.string(",").trim(_e.optWhitespace),e.Number,((e,t,i,n,s)=>new Ne({R:s,P:e,Y:i})));SimpleSerializationVector2D=e=>_e.seqMap(e.Number,_e.string(",").trim(_e.optWhitespace),e.Number,((e,t,i)=>new Ae({X:e,Y:i})));SimpleSerializationVector=e=>_e.seqMap(e.Number,_e.string(",").trim(_e.optWhitespace),e.Number,_e.string(",").trim(_e.optWhitespace),e.Number,((e,t,i,n,s)=>new Be({X:e,Y:i,Z:s})));LinearColor=e=>Fe.createEntityGrammar(e,Ce,!1);FunctionReference=e=>Fe.createEntityGrammar(e,ve);VariableReference=e=>Fe.createEntityGrammar(e,He);MacroGraphReference=e=>Fe.createEntityGrammar(e,Se);KeyBinding=e=>_e.alt(e.Identifier.map((e=>new we({Key:e}))),Fe.createEntityGrammar(e,we));Pin=e=>Fe.createEntityGrammar(e,De);CustomProperties=e=>_e.regex(/CustomProperties\s+/).then(e.Pin).map((e=>t=>{let i=ae.objectGet(t,["CustomProperties"],[]);i.push(e),ae.objectSet(t,["CustomProperties"],i,!0)}));Object=e=>_e.seqMap(_e.regexp(/Begin\s+Object\s+/),_e.alt(e.CustomProperties,Fe.createAttributeGrammar(e,Oe)).sepBy1(_e.whitespace),_e.regexp(/\s+End\s+Object/),((e,t,i)=>{let n={};return t.forEach((e=>e(n))),new Oe(n)}));MultipleObject=e=>e.Object.sepBy1(_e.whitespace).trim(_e.optWhitespace);LinearColorFromHex=e=>_e.string("#").then(e.HexDigit.times(2).tie().times(3,4)).trim(_e.optWhitespace).map((([e,t,i,n])=>new Ce({R:parseInt(e,16)/255,G:parseInt(t,16)/255,B:parseInt(i,16)/255,A:n?parseInt(n,16)/255:1})));LinearColorFromRGBList=e=>_e.seqMap(e.ColorNumber,_e.string(",").skip(_e.optWhitespace),e.ColorNumber,_e.string(",").skip(_e.optWhitespace),e.ColorNumber.map(Number),((e,t,i,n,s)=>new Ce({R:e/255,G:i/255,B:s/255,A:1})));LinearColorFromRGB=e=>_e.string("rgb").then(e.LinearColorFromRGBList.wrap(_e.regex(/\(\s*/),_e.regex(/\s*\)/)));LinearColorFromRGBA=e=>_e.string("rgba").then(_e.seqMap(e.ColorNumber,_e.string(",").skip(_e.optWhitespace),e.ColorNumber,_e.string(",").skip(_e.optWhitespace),e.ColorNumber.map(Number),_e.string(",").skip(_e.optWhitespace),_e.regex(/0?\.\d+|[01]/).map(Number),((e,t,i,n,s,r,a)=>new Ce({R:e/255,G:i/255,B:s/255,A:a}))).wrap(_e.regex(/\(\s*/),_e.regex(/\s*\)/)));LinearColorFromAnyColor=e=>_e.alt(e.LinearColorFromRGBList,e.LinearColorFromHex,e.LinearColorFromRGB,e.LinearColorFromRGBA);UnknownKeys=e=>_e.seqMap(_e.regex(/\w*\s*/).skip(_e.string("(")),_e.seqMap(e.AttributeName,_e.string("=").trim(_e.optWhitespace),e.AttributeAnyValue,((e,t,i)=>t=>ae.objectSet(t,e.split("."),i,!0))).trim(_e.optWhitespace).sepBy(_e.string(",")).skip(_e.regex(/,?/).then(_e.optWhitespace)),_e.string(")"),((e,t,i)=>{let n={};t.forEach((e=>e(n)));let s=new Re(n);return e&&(s.lookbehind=e),s}))}class Ge{static grammar=Ie.createLanguage(new Fe);constructor(e,t="",i=",",n=!1,s="=",r=(e=>e.join("."))){this.entityType=e,this.attributePrefix=t,this.attributeSeparator=i,this.trailingSeparator=n,this.attributeValueConjunctionSign=s,this.attributeKeyPrinter=r}deserialize(e){return this.read(e)}serialize(e,t=!1,i=e){return this.write(i,e,t)}read(e){throw new Error("Not implemented")}write(e,t,i){throw new Error("Not implemented")}writeValue(e,t,i,n){const s=ae.getType(t),r=ne.getSerializer(s);if(!r)throw new Error(`Unknown value type "${s.name}", a serializer must be registered in the SerializerFactory class, check initializeSerializerFactory.js`);return r.write(t instanceof oe?t:e,t,n)}subWrite(e,t,i,n){let s="",r=t.concat("");const a=r.length-1,o=i.constructor.attributes,l=o?ae.mergeArrays(Object.keys(o),Object.keys(i)):Object.keys(i);for(const t of l){r[a]=t;const o=i[t];if(o?.constructor===Object)s+=(s.length?this.attributeSeparator:"")+this.subWrite(e,r,o,n);else if(void 0!==o&&this.showProperty(e,i,r,o)){const t=ae.isSerialized(e,r);s+=(s.length?this.attributeSeparator:"")+this.attributePrefix+this.attributeKeyPrinter(r)+this.attributeValueConjunctionSign+(t?`"${this.writeValue(e,o,r,!0)}"`:this.writeValue(e,o,r,n))}}return this.trailingSeparator&&s.length&&1===r.length&&(s+=this.attributeSeparator),s}showProperty(e,t,i,n){const s=this.entityType.attributes,r=ae.objectGet(s,i);return r?.constructor!==Object||!r.ignored&&(!ae.equals(r.value,n)||r.showDefault)}}class je extends Ge{constructor(){super(Oe," ","\n",!1)}showProperty(e,t,i,n){switch(i.toString()){case"Class":case"Name":case"CustomProperties":return!1}return super.showProperty(e,t,i,n)}read(e){const t=Ge.grammar.Object.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}readMultiple(e){const t=Ge.grammar.MultipleObject.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}write(e,t,i){return`Begin Object Class=${t.Class.path} Name=${this.writeValue(e,t.Name,["Name"],i)}\n${this.subWrite(e,[],t,i)+t.CustomProperties.map((e=>this.attributeSeparator+this.attributePrefix+"CustomProperties "+ne.getSerializer(De).serialize(e))).join("")}\nEnd Object\n`}}class Ue extends ie{static#l=new je;#u;constructor(e,t,i={}){i.listenOnFocus??=!0,i.unlistenOnTextEdit??=!0,super(e,t,i);let n=this;this.#u=e=>n.copied()}listenEvents(){window.addEventListener("copy",this.#u)}unlistenEvents(){window.removeEventListener("copy",this.#u)}getSerializedText(){return this.blueprint.getNodes(!0).map((e=>Ue.#l.serialize(e.entity,!1))).join("")}copied(){const e=this.getSerializedText();navigator.clipboard.writeText(e)}}class Ke{element;get blueprint(){return this.element.blueprint}#c=[];get inputObjects(){return this.#c}initialize(e){this.element=e}createInputObjects(){return[]}getInputObject(e){return this.inputObjects.find((t=>t.constructor==e))}setup(){this.#c.forEach((e=>e.setup()))}cleanup(){this.#c.forEach((e=>e.cleanup()))}willUpdate(e){}update(e){}render(){return $``}firstUpdated(e){}updated(e){}inputSetup(){this.#c=this.createInputObjects()}}class We extends ie{#h;constructor(e,t,i={}){i.activateAnyKey??=!1,i.activationKeys??=[],i.consumeEvent??=!0,i.listenOnFocus??=!0,i.unlistenOnTextEdit??=!0,i.activationKeys instanceof Array||(i.activationKeys=[i.activationKeys]),i.activationKeys=i.activationKeys.map((e=>{if(e instanceof we)return e;if(e.constructor===String){const t=Ge.grammar.KeyBinding.parse(e);if(t.status)return t.value}throw new Error("Unexpected key value")})),super(e,t,i),this.#h=this.options.activationKeys??[];let n=this;this.keyDownHandler=e=>{(this.options.activateAnyKey||n.#h.some((t=>(e=>e.bShift||"LeftShift"==e.Key||"RightShift"==e.Key)(t)==e.shiftKey&&(e=>e.bCtrl||"LeftControl"==e.Key||"RightControl"==e.Key)(t)==e.ctrlKey&&(e=>e.bAlt||"LeftAlt"==e.Key||"RightAlt"==e.Key)(t)==e.altKey&&te.Keys[t.Key]==e.code)))&&(i.consumeEvent&&(e.preventDefault(),e.stopImmediatePropagation()),n.fire(),document.removeEventListener("keydown",n.keyDownHandler),document.addEventListener("keyup",n.keyUpHandler))},this.keyUpHandler=e=>{(this.options.activateAnyKey||n.#h.some((t=>t.bShift&&"Shift"==e.key||t.bCtrl&&"Control"==e.key||t.bAlt&&"Alt"==e.key||t.bCmd&&"Meta"==e.key||te.Keys[t.Key]==e.code)))&&(i.consumeEvent&&e.stopImmediatePropagation(),n.unfire(),document.removeEventListener("keyup",this.keyUpHandler),document.addEventListener("keydown",this.keyDownHandler))}}listenEvents(){document.addEventListener("keydown",this.keyDownHandler)}unlistenEvents(){document.removeEventListener("keydown",this.keyDownHandler)}fire(){}unfire(){}}class Ze extends We{constructor(e,t,i={}){i.activationKeys=te.deleteNodesKeyboardKey,super(e,t,i)}fire(){this.blueprint.removeGraphElement(...this.blueprint.getNodes(!0))}}class Xe extends ie{constructor(e,t,i={}){i.ignoreTranslateCompensate??=!1,i.ignoreScale??=!1,i.movementSpace??=t.getGridDOMElement()??document.documentElement,super(e,t,i),this.movementSpace=i.movementSpace}locationFromEvent(e){const t=ae.convertLocation([e.clientX,e.clientY],this.movementSpace,this.options.ignoreScale);return this.options.ignoreTranslateCompensate?t:this.blueprint.compensateTranslation(t[0],t[1])}}class Ye extends Xe{#d=e=>{e.preventDefault();const t=this.locationFromEvent(e);this.wheel(Math.sign(e.deltaY*te.mouseWheelFactor),t)};#p=e=>e.preventDefault();constructor(e,t,i={}){i.listenOnFocus=!0,i.strictTarget??=!1,super(e,t,i),this.strictTarget=i.strictTarget}listenEvents(){this.movementSpace.addEventListener("wheel",this.#d,!1),this.movementSpace.parentElement?.addEventListener("wheel",this.#p)}unlistenEvents(){this.movementSpace.removeEventListener("wheel",this.#d,!1),this.movementSpace.parentElement?.removeEventListener("wheel",this.#p)}wheel(e,t){}}class qe extends Ye{#m=!1;get enableZoonIn(){return this.#m}set enableZoonIn(e){e!=this.#m&&(this.#m=e)}wheel(e,t){let i=this.blueprint.getZoom();e=-e,!this.enableZoonIn&&0==i&&e>0||(i+=e,this.blueprint.setZoom(i,t))}}class Je extends We{#g;constructor(e,t,i={}){i.activationKeys=te.enableZoomIn,super(e,t,i)}fire(){this.#g=this.blueprint.getInputObject(qe),this.#g.enableZoonIn=!0}unfire(){this.#g.enableZoonIn=!1}}class Qe extends We{constructor(e,t,i={}){i.activationKeys=te.selectAllKeyboardKey,super(e,t,i)}fire(){this.blueprint.selectAll()}}class et extends Q{#b=[];#t;get blueprint(){return this.#t}set blueprint(e){this.#t=e}#v;get entity(){return this.#v}set entity(e){this.#v=e}#f;get template(){return this.#f}isInitialized=!1;isSetup=!1;inputObjects=[];initialize(e,t){this.requestUpdate(),this.#v=e,this.#f=t,this.#f.initialize(this),this.isConnected&&this.updateComplete.then((()=>this.setup())),this.isInitialized=!0}connectedCallback(){super.connectedCallback(),this.blueprint=this.closest("ueb-blueprint"),this.isInitialized&&(this.requestUpdate(),this.updateComplete.then((()=>this.setup())))}disconnectedCallback(){super.disconnectedCallback(),this.isSetup&&this.updateComplete.then((()=>this.cleanup())),this.acknowledgeDelete()}createRenderRoot(){return this}shouldUpdate(e){return this.isInitialized&&this.isConnected}setup(){this.template.setup(),this.isSetup=!0}cleanup(){this.template.cleanup(),this.isSetup=!1}willUpdate(e){super.willUpdate(e),this.template.willUpdate(e)}update(e){super.update(e),this.template.update(e)}render(){return this.template.render()}firstUpdated(e){super.firstUpdated(e),this.template.firstUpdated(e),this.template.inputSetup()}updated(e){super.updated(e),this.template.updated(e);for(const t of this.#b)t(e);this.#b=[]}addNextUpdatedCallbacks(e,t=!1){this.#b.push(e),t&&this.requestUpdate()}acknowledgeDelete(){let e=new CustomEvent(te.removeEventName);this.dispatchEvent(e)}isSameGraph(e){return this.blueprint&&this.blueprint==e?.blueprint}getInputObject(e){return this.template.inputObjects.find((t=>t.constructor==e))}}class tt extends et{static properties={...super.properties,locationX:{type:Number,attribute:!1},locationY:{type:Number,attribute:!1},sizeX:{type:Number,attribute:!1},sizeY:{type:Number,attribute:!1}};static dragEventName=te.dragEventName;static dragGeneralEventName=te.dragGeneralEventName;constructor(){super(),this.locationX=0,this.locationY=0,this.sizeX=0,this.sizeY=0}computeSizes(){const e=this.getBoundingClientRect();this.sizeX=this.blueprint.scaleCorrect(e.width),this.sizeY=this.blueprint.scaleCorrect(e.height)}firstUpdated(e){super.firstUpdated(e),this.computeSizes()}setLocation(e,t,i=!0){const n=e-this.locationX,s=t-this.locationY;if(this.locationX=e,this.locationY=t,this.blueprint&&i){const e=new CustomEvent(this.constructor.dragEventName,{detail:{value:[n,s]},bubbles:!1,cancelable:!0});this.dispatchEvent(e)}}addLocation(e,t,i=!0){this.setLocation(this.locationX+e,this.locationY+t,i)}acknowledgeDrag(e){const t=new CustomEvent(this.constructor.dragGeneralEventName,{detail:{value:e},bubbles:!0,cancelable:!0});this.dispatchEvent(t)}snapToGrid(){const e=ae.snapToGrid(this.locationX,this.locationY,te.gridSize);this.locationX==e[0]&&this.locationY==e[1]||this.setLocation(e[0],e[1])}topBoundary(e=!1){return this.template.topBoundary(e)}rightBoundary(e=!1){return this.template.rightBoundary(e)}bottomBoundary(e=!1){return this.template.bottomBoundary(e)}leftBoundary(e=!1){return this.template.leftBoundary(e)}}class it extends Xe{#y=e=>{if(this.blueprint.setFocused(!0),e.button===this.options.clickButton)this.options.strictTarget&&e.target!=e.currentTarget||(this.options.consumeEvent&&e.stopImmediatePropagation(),this.#w.addEventListener("mousemove",this.#E),document.addEventListener("mouseup",this.#C),this.clickedPosition=this.locationFromEvent(e),this.blueprint.mousePosition[0]=this.clickedPosition[0],this.blueprint.mousePosition[1]=this.clickedPosition[1],this.target instanceof tt&&(this.clickedOffset=[this.clickedPosition[0]-this.target.locationX,this.clickedPosition[1]-this.target.locationY]),this.clicked(this.clickedPosition));else this.options.exitAnyButton||this.#C(e)};#E=e=>{this.options.consumeEvent&&e.stopImmediatePropagation(),this.#w.removeEventListener("mousemove",this.#E),this.#w.addEventListener("mousemove",this.#S);const t=this.getEvent(te.trackingMouseEventName.begin);this.#x=0==this.target.dispatchEvent(t);const i=this.locationFromEvent(e);this.lastLocation=ae.snapToGrid(this.clickedPosition[0],this.clickedPosition[1],this.stepSize),this.startDrag(i),this.started=!0};#S=e=>{this.options.consumeEvent&&e.stopImmediatePropagation();const t=this.locationFromEvent(e),i=[e.movementX,e.movementY];if(this.dragTo(t,i),this.#x&&(this.blueprint.mousePosition=t),this.options.scrollGraphEdge){const e=Math.sqrt(i[0]*i[0]+i[1]*i[1]),n=this.blueprint.scaleCorrect(te.edgeScrollThreshold),s=this.blueprint.template.gridLeftVisibilityBoundary()+n,r=this.blueprint.template.gridRightVisibilityBoundary()-n;let a=0;t[0]r&&(a=t[0]-r);const o=this.blueprint.template.gridTopVisibilityBoundary()+n,l=this.blueprint.template.gridBottomVisibilityBoundary()-n;let u=0;t[1]l&&(u=t[1]-l),a=ae.clamp(this.blueprint.scaleCorrectReverse(a)**3*e*.6,-20,20),u=ae.clamp(this.blueprint.scaleCorrectReverse(u)**3*e*.6,-20,20),this.blueprint.scrollDelta(a,u)}};#C=e=>{if(!this.options.exitAnyButton||e.button==this.options.clickButton){if(this.options.consumeEvent&&e.stopImmediatePropagation(),this.#w.removeEventListener("mousemove",this.#E),this.#w.removeEventListener("mousemove",this.#S),document.removeEventListener("mouseup",this.#C),this.started&&this.endDrag(),this.unclicked(),this.#x){const e=this.getEvent(te.trackingMouseEventName.end);this.target.dispatchEvent(e),this.#x=!1}this.started=!1}};#x=!1;#w;#P;clickedOffset=[0,0];clickedPosition=[0,0];lastLocation=[0,0];started=!1;stepSize=1;constructor(e,t,i={}){i.clickButton??=0,i.consumeEvent??=!0,i.draggableElement??=e,i.exitAnyButton??=!0,i.moveEverywhere??=!1,i.movementSpace??=t?.getGridDOMElement(),i.repositionOnClick??=!1,i.scrollGraphEdge??=!1,i.strictTarget??=!1,super(e,t,i),this.stepSize=parseInt(i?.stepSize??te.gridSize),this.#w=this.options.moveEverywhere?document.documentElement:this.movementSpace,this.#P=this.options.draggableElement,this.listenEvents()}listenEvents(){super.listenEvents(),this.#P.addEventListener("mousedown",this.#y),2==this.options.clickButton&&this.#P.addEventListener("contextmenu",(e=>e.preventDefault()))}unlistenEvents(){super.unlistenEvents(),this.#P.removeEventListener("mousedown",this.#y)}getEvent(e){return new CustomEvent(e,{detail:{tracker:this},bubbles:!0,cancelable:!0})}clicked(e){}startDrag(e){}dragTo(e,t){}endDrag(){}unclicked(e){}}class nt extends it{startDrag(){this.blueprint.scrolling=!0}dragTo(e,t){this.blueprint.scrollDelta(-t[0],-t[1])}endDrag(){this.blueprint.scrolling=!1}}class st extends Xe{#k=null;#L=e=>{e.preventDefault(),this.blueprint.mousePosition=this.locationFromEvent(e)};#N=e=>{this.#k||(e.preventDefault(),this.#k=e.detail.tracker,this.unlistenMouseMove())};#T=e=>{this.#k==e.detail.tracker&&(e.preventDefault(),this.#k=null,this.listenMouseMove())};constructor(e,t,i={}){i.listenOnFocus=!0,super(e,t,i)}listenMouseMove(){this.target.addEventListener("mousemove",this.#L)}unlistenMouseMove(){this.target.removeEventListener("mousemove",this.#L)}listenEvents(){this.listenMouseMove(),this.blueprint.addEventListener(te.trackingMouseEventName.begin,this.#N),this.blueprint.addEventListener(te.trackingMouseEventName.end,this.#T)}unlistenEvents(){this.unlistenMouseMove(),this.blueprint.removeEventListener(te.trackingMouseEventName.begin,this.#N),this.blueprint.removeEventListener(te.trackingMouseEventName.end,this.#T)}}class rt{static#A=new Map;static registerElement(e,t){rt.#A.set(e,t)}static getConstructor(e){return rt.#A.get(e)}}class at extends ie{static#l=new je;#M;constructor(e,t,i={}){i.listenOnFocus??=!0,i.unlistenOnTextEdit??=!0,super(e,t,i);let n=this;this.#M=e=>n.pasted(e.clipboardData.getData("Text"))}listenEvents(){window.addEventListener("paste",this.#M)}unlistenEvents(){window.removeEventListener("paste",this.#M)}pasted(e){let t=0,i=0,n=0,s=at.#l.readMultiple(e).map((e=>{let s=rt.getConstructor("ueb-node").newObject(e);return t+=s.locationY,i+=s.locationX,++n,s}));t/=n,i/=n,s.length>0&&this.blueprint.unselectAll();let r=this.blueprint.mousePosition;return s.forEach((e=>{e.addLocation(r[0]-i,r[1]-t),e.snapToGrid(),e.setSelected(!0)})),this.blueprint.addGraphElement(...s),!0}}class ot extends it{constructor(e,t,i={}){i.scrollGraphEdge??=!0,super(e,t,i),this.selectorElement=this.blueprint.template.selectorElement}startDrag(){this.selectorElement.beginSelect(this.clickedPosition)}dragTo(e,t){this.selectorElement.selectTo(e)}endDrag(){this.started&&this.selectorElement.endSelect()}unclicked(){this.started||this.blueprint.unselectAll()}}class lt extends ie{#B=e=>this.clickedSomewhere(e.target);constructor(e,t,i={}){i.listenOnFocus=!0,super(e,t,i),this.blueprint.focus&&document.addEventListener("click",this.#B)}clickedSomewhere(e){e.closest("ueb-blueprint")||this.blueprint.setFocused(!1)}listenEvents(){document.addEventListener("click",this.#B)}unlistenEvents(){document.removeEventListener("click",this.#B)}}class ut extends Ke{static styleVariables={"--ueb-font-size":`${te.fontSize}`,"--ueb-grid-axis-line-color":`${te.gridAxisLineColor}`,"--ueb-grid-expand":`${te.expandGridSize}px`,"--ueb-grid-line-color":`${te.gridLineColor}`,"--ueb-grid-line-width":`${te.gridLineWidth}px`,"--ueb-grid-set-line-color":`${te.gridSetLineColor}`,"--ueb-grid-set":`${te.gridSet}`,"--ueb-grid-size":`${te.gridSize}px`,"--ueb-link-min-width":`${te.linkMinWidth}`,"--ueb-node-radius":`${te.nodeRadius}px`};#D=new ResizeObserver((e=>{const t=e.find((e=>e.target===this.viewportElement))?.devicePixelContentBoxSize?.[0];t&&(this.viewportSize[0]=t.inlineSize,this.viewportSize[1]=t.blockSize)}));headerElement;overlayElement;viewportElement;selectorElement;gridElement;linksContainerElement;nodesContainerElement;viewportSize=[0,0];initialize(e){super.initialize(e),this.element.style.cssText=Object.entries(ut.styleVariables).map((([e,t])=>`${e}:${t};`)).join("")}setup(){super.setup(),this.#D.observe(this.viewportElement,{box:"device-pixel-content-box"});const e=this.viewportElement.getBoundingClientRect();this.viewportSize[0]=e.width,this.viewportSize[1]=e.height,this.blueprint.nodes.length>0&&(this.blueprint.requestUpdate(),this.blueprint.updateComplete.then((()=>this.centerContentInViewport())))}cleanup(){super.cleanup(),this.#D.unobserve(this.viewportElement)}createInputObjects(){return[...super.createInputObjects(),new Ue(this.element.getGridDOMElement(),this.element),new at(this.element.getGridDOMElement(),this.element),new Ze(this.element.getGridDOMElement(),this.element),new Qe(this.element.getGridDOMElement(),this.element),new qe(this.element.getGridDOMElement(),this.element),new ot(this.element.getGridDOMElement(),this.element,{clickButton:0,exitAnyButton:!0,moveEverywhere:!0}),new nt(this.element.getGridDOMElement(),this.element,{clickButton:2,exitAnyButton:!1,moveEverywhere:!0}),new lt(this.element.getGridDOMElement(),this.element),new st(this.element.getGridDOMElement(),this.element),new Je(this.element.getGridDOMElement(),this.element)]}render(){return $`
Zoom ${0==this.element.zoom?"1:1":(this.element.zoom>0?"+":"")+this.element.zoom}
`}firstUpdated(e){super.firstUpdated(e),this.headerElement=this.element.querySelector(".ueb-viewport-header"),this.overlayElement=this.element.querySelector(".ueb-viewport-overlay"),this.viewportElement=this.element.querySelector(".ueb-viewport-body"),this.selectorElement=this.element.querySelector("ueb-selector"),this.gridElement=this.viewportElement.querySelector(".ueb-grid"),this.linksContainerElement=this.element.querySelector("[data-links]"),this.linksContainerElement.append(...this.element.getLinks()),this.nodesContainerElement=this.element.querySelector("[data-nodes]"),this.nodesContainerElement.append(...this.element.getNodes()),this.viewportElement.scroll(te.expandGridSize,te.expandGridSize)}willUpdate(e){super.willUpdate(e),this.headerElement&&e.has("zoom")&&(this.headerElement.classList.add("ueb-zoom-changed"),this.headerElement.addEventListener("animationend",(()=>this.headerElement.classList.remove("ueb-zoom-changed"))))}updated(e){if(super.updated(e),(e.has("scrollX")||e.has("scrollY"))&&this.viewportElement.scroll(this.element.scrollX,this.element.scrollY),e.has("zoom")){this.element.style.setProperty("--ueb-scale",this.blueprint.getScale());const t=e.get("zoom"),i=Math.min(t,this.element.zoom),n=Math.max(t,this.element.zoom),s=ae.range(i,n),r=e=>`ueb-zoom-${e}`;te<0)).map(r)),this.element.classList.add(...s.filter((e=>e>0)).map(r))):(this.element.classList.remove(...s.filter((e=>e>0)).map(r)),this.element.classList.add(...s.filter((e=>e<0)).map(r)))}}getCommentNodes(e=!1){return this.element.querySelectorAll(`ueb-node[data-type="${te.nodeType.comment}"]${e?'[data-selected="true"]':""}`)}getPin(e){return this.element.querySelector(`ueb-node[data-name="${e.objectName}"] ueb-pin[data-id="${e.pinGuid}"]`)}getCopyInputObject(){return this.getInputObject(Ue)}isPointVisible(e,t){return!1}gridTopVisibilityBoundary(){return this.blueprint.scaleCorrect(this.blueprint.scrollY)-this.blueprint.translateY}gridRightVisibilityBoundary(){return this.blueprint,this.gridLeftVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[0])}gridBottomVisibilityBoundary(){return this.gridTopVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[1])}gridLeftVisibilityBoundary(){return this.blueprint.scaleCorrect(this.blueprint.scrollX)-this.blueprint.translateX}centerViewport(e=0,t=0,i=!0){const n=this.gridLeftVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[0]/2),s=this.gridTopVisibilityBoundary()+this.blueprint.scaleCorrect(this.viewportSize[1]/2);this.blueprint.scrollDelta(this.blueprint.scaleCorrectReverse(e-n),this.blueprint.scaleCorrectReverse(t-s),i)}centerContentInViewport(e=!0){let t=0,i=0;const n=this.blueprint.getNodes();for(const e of n)t+=e.leftBoundary()+e.rightBoundary(),i+=e.topBoundary()+e.bottomBoundary();t=n.length>0?Math.round(t/(2*n.length)):0,i=n.length>0?Math.round(i/(2*n.length)):0,this.centerViewport(t,i,e)}}class ct extends et{static properties={...super.properties,fromX:{type:Number,attribute:!1},fromY:{type:Number,attribute:!1},toX:{type:Number,attribute:!1},toY:{type:Number,attribute:!1}};constructor(){super(),this.fromX=0,this.fromY=0,this.toX=0,this.toY=0}setBothLocations([e,t]){this.fromX=e,this.fromY=t,this.toX=e,this.toY=t}addSourceLocation(e,t){this.fromX+=e,this.fromY+=t}addDestinationLocation(e,t){this.toX+=e,this.toY+=t}}class ht extends Ke{update(e){super.update(e);const[t,i,n,s]=[Math.round(this.element.fromX),Math.round(this.element.fromY),Math.round(this.element.toX),Math.round(this.element.toY)],[r,a,o,l]=[Math.min(t,n),Math.min(i,s),Math.abs(t-n),Math.abs(i-s)];(e.has("fromX")||e.has("toX"))&&(this.element.style.left=`${r}px`,this.element.style.width=`${o}px`),(e.has("fromY")||e.has("toY"))&&(this.element.style.top=`${a}px`,this.element.style.height=`${l}px`)}}class dt extends Oe{constructor(e={},t){super(e,!0),this.Class=new be("/Script/BlueprintGraph.K2Node_Knot"),this.Name="K2Node_Knot";const i=new De({PinName:"InputPin"},!0),n=new De({PinName:"OutputPin",Direction:"EGPD_Output"},!0);t&&(i.copyTypeFrom(t),n.copyTypeFrom(t)),this.CustomProperties=[i,n]}}class pt extends Xe{static ignoreDbClick=e=>{};#$=e=>{this.options.strictTarget&&e.target!==e.currentTarget||(this.options.consumeEvent&&e.stopImmediatePropagation(),this.clickedPosition=this.locationFromEvent(e),this.blueprint.mousePosition[0]=this.clickedPosition[0],this.blueprint.mousePosition[1]=this.clickedPosition[1],this.dbclicked(this.clickedPosition))};#H;get onDbClick(){return this.#H}set onDbClick(e){this.#H=e}clickedPosition=[0,0];constructor(e,t,i={},n=pt.ignoreDbClick){i.consumeEvent??=!0,i.strictTarget??=!1,super(e,t,i),this.#H=n,this.listenEvents()}listenEvents(){this.target.addEventListener("dblclick",this.#$)}unlistenEvents(){this.target.removeEventListener("dblclick",this.#$)}dbclicked(e){this.onDbClick(e)}}class mt extends ht{static decreasingValue(e,t){const i=-e*t[0]**2,n=t[1]-i/t[0];return e=>i/e+n}static clampedLine(e,t){if(e[0]>t[0]){const i=e;e=t,t=i}const i=(t[1]-e[1])/(t[0]-e[0]),n=e[1]-i*e[0];return s=>st[0]?t[1]:i*s+n}static c1DecreasingValue=mt.decreasingValue(-.15,[100,15]);static c2DecreasingValue=mt.decreasingValue(-.05,[500,130]);static c2Clamped=mt.clampedLine([0,80],[200,40]);#O=`ueb-id-${Math.floor(1e12*Math.random())}`;#V=e=>{const t=new dt({},this.element.sourcePin.entity),i=rt.getConstructor("ueb-node").newObject(t);i.setLocation(...this.blueprint.snapToGrid(...e));const n=i.template;this.blueprint.addGraphElement(i);const s=this.element.getInputPin(),r=this.element.getOutputPin();this.element.sourcePin=null,this.element.destinationPin=null;const a=rt.getConstructor("ueb-link").newObject(r,n.inputPin);this.blueprint.addGraphElement(a),this.element.sourcePin=n.outputPin,this.element.destinationPin=s};createInputObjects(){return[...super.createInputObjects(),new pt(this.element.querySelector(".ueb-link-area"),this.blueprint,void 0,(e=>{e[0]+=te.knotOffset[0],e[1]+=te.knotOffset[1],this.#V(e)}))]}willUpdate(e){super.willUpdate(e);const t=this.element.sourcePin,i=this.element.destinationPin;if(e.has("fromX")||e.has("toX")){const e=this.element.fromX,n=this.element.toX,s=t?.nodeElement.getType()==te.nodeType.knot,r=i?.nodeElement.getType()==te.nodeType.knot;!s||i&&!r||(t?.isInput()&&n>e+te.distanceThreshold?this.element.sourcePin=t.nodeElement.template.outputPin:t?.isOutput()&&ne+te.distanceThreshold&&(this.element.destinationPin=i.nodeElement.template.inputPin))}const n=Math.max(Math.abs(this.element.fromX-this.element.toX),1),s=Math.max(Math.abs(this.element.fromY-this.element.toY),1),r=Math.max(n,te.linkMinWidth),a=n/r,o=this.element.originatesFromInput?this.element.fromXthis.element.toY?1:0)),this.element.style.setProperty("--ueb-start-percentage",`${Math.round(this.element.startPercentage)}%`),this.element.style.setProperty("--ueb-link-start",`${Math.round(this.element.startPixels)}`)}render(){return $` ${this.element.linkMessageIcon||this.element.linkMessageText?$``:O}`}}class gt extends ct{static properties={...super.properties,source:{type:String,reflect:!0},destination:{type:String,reflect:!0},dragging:{type:Boolean,attribute:"data-dragging",converter:ae.booleanConverter,reflect:!0},originatesFromInput:{type:Boolean,attribute:!1},svgPathD:{type:String,attribute:!1},linkMessageIcon:{type:String,attribute:!1},linkMessageText:{type:String,attribute:!1}};#z;get sourcePin(){return this.#z}set sourcePin(e){this.#I(e,!1)}#R;get destinationPin(){return this.#R}set destinationPin(e){this.#I(e,!0)}#_=()=>this.remove();#F=e=>this.addSourceLocation(...e.detail.value);#G=e=>this.addDestinationLocation(...e.detail.value);#j=e=>this.setSourceLocation();#U=e=>this.setDestinationLocation();linkMessageIcon=O;linkMessageText=O;pathElement;constructor(){super(),this.source=null,this.destination=null,this.dragging=!1,this.originatesFromInput=!1,this.startPercentage=0,this.svgPathD="",this.startPixels=0}static newObject(e,t){const i=new gt;return i.initialize(e,t),i}initialize(e,t){super.initialize({},new mt),e&&(this.sourcePin=e,t||(this.toX=this.fromX,this.toY=this.fromY)),t&&(this.destinationPin=t,e||(this.fromX=this.toX,this.fromY=this.toY))}#I(e,t){const i=()=>t?this.destinationPin:this.sourcePin;if(i()!=e){if(i()){const e=i().getNodeElement();e.removeEventListener(te.removeEventName,this.#_),e.removeEventListener(te.nodeDragEventName,t?this.#G:this.#F),e.removeEventListener(te.nodeReflowEventName,t?this.#U:this.#j),this.#K()}if(t?this.#R=e:this.#z=e,i()){const e=i().getNodeElement();e.addEventListener(te.removeEventName,this.#_),e.addEventListener(te.nodeDragEventName,t?this.#G:this.#F),e.addEventListener(te.nodeReflowEventName,t?this.#U:this.#j),t?this.setDestinationLocation():(this.setSourceLocation(),this.originatesFromInput=this.sourcePin.isInput()),this.#W()}}}#W(){this.sourcePin&&this.destinationPin&&(this.sourcePin.linkTo(this.destinationPin),this.destinationPin.linkTo(this.sourcePin))}#K(){this.sourcePin&&this.destinationPin&&(this.sourcePin.unlinkFrom(this.destinationPin,!1),this.destinationPin.unlinkFrom(this.sourcePin,!1))}cleanup(){super.cleanup(),this.#K(),this.sourcePin=null,this.destinationPin=null}setSourceLocation(e=null,t=!0){if(null==e){const i=this;if(t&&(!this.hasUpdated||!this.sourcePin.hasUpdated))return void Promise.all([this.updateComplete,this.sourcePin.updateComplete]).then((()=>i.setSourceLocation(null,!1)));e=this.sourcePin.template.getLinkLocation()}const[i,n]=e;this.fromX=i,this.fromY=n}setDestinationLocation(e=null,t=!0){if(null==e){const i=this;if(t&&(!this.hasUpdated||!this.destinationPin.hasUpdated))return void Promise.all([this.updateComplete,this.destinationPin.updateComplete]).then((()=>i.setDestinationLocation(null,!1)));e=this.destinationPin.template.getLinkLocation()}this.toX=e[0],this.toY=e[1]}getInputPin(){return this.sourcePin?.isInput()?this.sourcePin:this.destinationPin}setInputPin(e){this.sourcePin?.isInput()&&(this.sourcePin=e),this.destinationPin=e}getOutputPin(){return this.destinationPin?.isOutput()?this.destinationPin:this.sourcePin}setOutputPin(e){this.destinationPin?.isOutput()&&(this.destinationPin=e),this.sourcePin=e}startDragging(){this.dragging=!0}finishDragging(){this.dragging=!1}removeMessage(){this.linkMessageIcon=O,this.linkMessageText=O}setMessageConvertType(){this.linkMessageIcon="ueb-icon-conver-type",this.linkMessageText=`Convert ${this.sourcePin.pinType} to ${this.destinationPin.pinType}.`}setMessageCorrect(){this.linkMessageIcon=$e.correct,this.linkMessageText=O}setMessageReplace(){this.linkMessageIcon=$e.correct,this.linkMessageText=O}setMessageDirectionsIncompatible(){this.linkMessageIcon=$e.reject,this.linkMessageText=$`Directions are not compatbile.`}setMessagePlaceNode(){this.linkMessageIcon="ueb-icon-place-node",this.linkMessageText=$`Place a new node.`}setMessageReplaceLink(){this.linkMessageIcon=$e.correct,this.linkMessageText=$`Replace existing input connections.`}setMessageReplaceOutputLink(){this.linkMessageIcon=$e.correct,this.linkMessageText=$`Replace existing output connections.`}setMessageSameNode(){this.linkMessageIcon=$e.reject,this.linkMessageText=$`Both are on the same node.`}setMEssagetypesIncompatible(){this.linkMessageIcon=$e.reject,this.linkMessageText=$`${this.sourcePin.pinType} is not compatible with ${this.destinationPin.pinType}.`}}class bt extends it{clicked(e){this.options.repositionOnClick&&(this.target.setLocation(...this.stepSize>1?ae.snapToGrid(e[0],e[1],this.stepSize):e),this.clickedOffset=[0,0])}dragTo(e,t){const i=[this.target.locationX??this.lastLocation[0],this.target.locationY??this.lastLocation[1]],[n,s]=this.stepSize>1?[ae.snapToGrid(e[0],e[1],this.stepSize),ae.snapToGrid(i[0],i[1],this.stepSize)]:[e,i];0==(t=[n[0]-this.lastLocation[0],n[1]-this.lastLocation[1]])[0]&&0==t[1]||(t[0]+=s[0]-i[0],t[1]+=s[1]-i[1],this.dragAction(n,t),this.lastLocation=n)}dragAction(e,t){this.target.setLocation(e[0]-this.clickedOffset[0],e[1]-this.clickedOffset[1])}}class vt extends bt{#Z;#X;#Y;#q;constructor(e,t,i={}){super(e,t,i),i.onClicked&&(this.#Z=i.onClicked),i.onStartDrag&&(this.#X=i.onStartDrag),i.onDrag&&(this.#Y=i.onDrag),i.onEndDrag&&(this.#q=i.onEndDrag)}clicked(e){super.clicked(e),this.#Z?.()}startDrag(){super.startDrag(),this.#X?.()}dragAction(e,t){this.#Y?.(e,t)}endDrag(){super.endDrag(),this.#q?.()}}class ft extends Ke{getDraggableElement(){return this.element}createDraggableObject(){return new bt(this.element,this.blueprint,{draggableElement:this.getDraggableElement()})}createInputObjects(){return[...super.createInputObjects(),this.createDraggableObject()]}topBoundary(e=!1){return this.element.locationY}rightBoundary(e=!1){return this.element.locationX+this.element.sizeX}bottomBoundary(e=!1){return this.element.locationY+this.element.sizeY}leftBoundary(e=!1){return this.element.locationX}centerInViewport(){const e=Math.min(this.blueprint.template.viewportSize[0]/10,this.blueprint.template.viewportSize[1]/10),t=this.leftBoundary()-this.blueprint.template.gridLeftVisibilityBoundary(),i=this.blueprint.template.gridRightVisibilityBoundary()-this.rightBoundary();let n=Math.max((t+i)/2,e);const s=this.topBoundary()-this.blueprint.template.gridTopVisibilityBoundary(),r=this.blueprint.template.gridBottomVisibilityBoundary()-this.bottomBoundary();let a=Math.max((s+r)/2,e);this.blueprint.scrollDelta(t-n,s-a,!0)}}class yt extends ft{update(e){super.update(e),e.has("locationX")&&(this.element.style.left=`${this.element.locationX}px`),e.has("locationY")&&(this.element.style.top=`${this.element.locationY}px`)}}class wt extends bt{startDrag(){this.target.selected||(this.blueprint.unselectAll(),this.target.setSelected(!0))}dragAction(e,t){this.target.acknowledgeDrag(t)}unclicked(){this.started?(this.blueprint.getNodes(!0).forEach((e=>e.boundComments.filter((t=>!e.isInsideComment(t))).forEach((t=>e.unbindFromComment(t))))),this.blueprint.getCommentNodes().forEach((e=>e.template.manageNodesBind()))):(this.blueprint.unselectAll(),this.target.setSelected(!0))}}class Et extends yt{getDraggableElement(){return this.element}createDraggableObject(){return new wt(this.element,this.blueprint,{draggableElement:this.getDraggableElement(),scrollGraphEdge:!0})}firstUpdated(e){super.firstUpdated(e),this.element.selected&&!this.element.listeningDrag&&this.element.setSelected(!0)}}class Ct extends Et{hasSubtitle=!1;static nodeStyleClasses=["ueb-node-style-default"];toggleAdvancedDisplayHandler=()=>{this.element.toggleShowAdvancedPinDisplay(),this.element.addNextUpdatedCallbacks((()=>this.element.acknowledgeReflow()),!0)};initialize(e){super.initialize(e),this.element.classList.add(...this.constructor.nodeStyleClasses),this.element.style.setProperty("--ueb-node-color",this.getColor().cssText)}getColor(){return this.element.entity.nodeColor()}render(){return $`
${this.renderTop()}
${this.element.entity.isDevelopmentOnly()?$`
Development Only
`:O} ${this.element.advancedPinDisplay?$`
${$e.expandIcon}
`:O}
`}renderNodeIcon(){return this.element.entity.nodeIcon()}renderNodeName(){return this.element.getNodeDisplayName()}renderTop(){const e=this.renderNodeIcon(),t=this.renderNodeName();return $`
${e?$`
${e}
`:O} ${t?$`
${t} ${this.hasSubtitle&&this.getTargetType().length>0?$`
Target is ${ae.formatStringName(this.getTargetType())}
`:O}
`:O}
`}firstUpdated(e){super.firstUpdated(e),this.setupPins(),this.element.updateComplete.then((()=>this.element.acknowledgeReflow()))}setupPins(){const e=this.element.querySelector(".ueb-node-inputs"),t=this.element.querySelector(".ueb-node-outputs");this.element.nodeNameElement=this.element.querySelector(".ueb-node-name-text");let i=!1,n=!1;this.element.getPinElements().forEach((s=>{s.isInput()?(e.appendChild(s),i=!0):s.isOutput()&&(t.appendChild(s),n=!0)})),i&&this.element.classList.add("ueb-node-has-inputs"),n&&this.element.classList.add("ueb-node-has-outputs")}createPinElements(){return this.element.getPinEntities().filter((e=>!e.isHidden())).map((e=>(this.hasSubtitle=this.hasSubtitle||"self"===e.PinName&&"Target"===e.getDisplayName(),rt.getConstructor("ueb-pin").newObject(e,void 0,this.element))))}getTargetType(){return this.element.entity.FunctionReference?.MemberParent?.getName()??"Untitled"}getPinElements(e){return e.querySelectorAll("ueb-pin")}linksChanged(){}}class St extends Ct{#J=document.createElement("div");#Q=document.createElement("div");#ee=document.createElement("div");#te=document.createElement("div");#ie=document.createElement("div");#ne=document.createElement("div");#se=document.createElement("div");#re=document.createElement("div");initialize(e){super.initialize(e),this.element.classList.add("ueb-resizeable"),this.#J.classList.add("ueb-resizeable-top"),this.#Q.classList.add("ueb-resizeable-right"),this.#ee.classList.add("ueb-resizeable-bottom"),this.#te.classList.add("ueb-resizeable-left"),this.#ie.classList.add("ueb-resizeable-top-right"),this.#ne.classList.add("ueb-resizeable-bottom-right"),this.#se.classList.add("ueb-resizeable-bottom-left"),this.#re.classList.add("ueb-resizeable-top-left")}update(e){super.update(e),this.element.sizeX>=0&&e.has("sizeX")&&(this.element.style.width=`${this.element.sizeX}px`),this.element.sizeY>=0&&e.has("sizeY")&&(this.element.style.height=`${this.element.sizeY}px`)}firstUpdated(e){super.firstUpdated(e),this.element.append(this.#J,this.#Q,this.#ee,this.#te,this.#ie,this.#ne,this.#se,this.#re)}createInputObjects(){return[...super.createInputObjects(),new vt(this.#J,this.blueprint,{onDrag:(e,t)=>{t[1]=e[1]-this.element.topBoundary(),this.setSizeY(this.element.sizeY-t[1])&&this.element.addLocation(0,t[1],!1)},onEndDrag:()=>this.endResize()}),new vt(this.#Q,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.rightBoundary(),this.setSizeX(this.element.sizeX+t[0])},onEndDrag:()=>this.endResize()}),new vt(this.#ee,this.blueprint,{onDrag:(e,t)=>{t[1]=e[1]-this.element.bottomBoundary(),this.setSizeY(this.element.sizeY+t[1])},onEndDrag:()=>this.endResize()}),new vt(this.#te,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.leftBoundary(),this.setSizeX(this.element.sizeX-t[0])&&this.element.addLocation(t[0],0,!1)},onEndDrag:()=>this.endResize()}),new vt(this.#ie,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.rightBoundary(),t[1]=e[1]-this.element.topBoundary(),this.setSizeX(this.element.sizeX+t[0]),this.setSizeY(this.element.sizeY-t[1])&&this.element.addLocation(0,t[1],!1)},onEndDrag:()=>this.endResize()}),new vt(this.#ne,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.rightBoundary(),t[1]=e[1]-this.element.bottomBoundary(),this.setSizeX(this.element.sizeX+t[0]),this.setSizeY(this.element.sizeY+t[1])},onEndDrag:()=>this.endResize()}),new vt(this.#se,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.leftBoundary(),t[1]=e[1]-this.element.bottomBoundary(),this.setSizeX(this.element.sizeX-t[0])&&this.element.addLocation(t[0],0,!1),this.setSizeY(this.element.sizeY+t[1])},onEndDrag:()=>this.endResize()}),new vt(this.#re,this.blueprint,{onDrag:(e,t)=>{t[0]=e[0]-this.element.leftBoundary(),t[1]=e[1]-this.element.topBoundary(),this.setSizeX(this.element.sizeX-t[0])&&this.element.addLocation(t[0],0,!1),this.setSizeY(this.element.sizeY-t[1])&&this.element.addLocation(0,t[1],!1)},onEndDrag:()=>this.endResize()})]}setSizeX(e){return this.element.setNodeWidth(e),!0}setSizeY(e){return this.element.setNodeHeight(e),!0}endResize(){}}class xt extends St{#ae=Ce.getWhite();#oe=0;initialize(e){e.entity.CommentColor&&(this.#ae.setFromRGBANumber(e.entity.CommentColor.toNumber()),this.#ae.setFromHSVA(this.#ae.H.value,this.#ae.S.value,.67*Math.pow(this.#ae.V.value,.45))),e.classList.add("ueb-node-style-comment","ueb-node-resizeable"),e.sizeX=25*te.gridSize,e.sizeY=6*te.gridSize,super.initialize(e)}getColor(){return r`${Math.round(255*this.#ae.R.value)}${Math.round(255*this.#ae.G.value)}${Math.round(255*this.#ae.B.value)}`}getDraggableElement(){return this.element.querySelector(".ueb-node-top")}render(){return $`
${this.element.entity.NodeComment}
`}firstUpdated(e){super.firstUpdated(e);const t=this.getDraggableElement().getBoundingClientRect();this.#oe=t.height}manageNodesBind(){let e=this.blueprint.getNodes();for(let t of e)t.topBoundary()>=this.element.topBoundary()&&t.rightBoundary()<=this.element.rightBoundary()&&t.bottomBoundary()<=this.element.bottomBoundary()&&t.leftBoundary()>=this.element.leftBoundary()?t.bindToComment(this.element):t.unbindFromComment(this.element)}setSizeX(e){return(e=Math.round(e))>=te.gridSet*te.gridSize&&(this.element.setNodeWidth(e),!0)}setSizeY(e){return(e=Math.round(e))>=3*te.gridSize&&(this.element.setNodeHeight(e),!0)}endResize(){this.manageNodesBind()}topBoundary(e=!1){return this.element.locationY}rightBoundary(e=!1){return this.element.locationX+this.element.sizeX}bottomBoundary(e=!1){return e?this.element.locationY+this.#oe:super.bottomBoundary()}leftBoundary(e=!1){return this.element.locationX}}class Pt extends it{#le;#ue=null;#ce=e=>{if(!this.enteredPin){this.linkValid=!1,this.enteredPin=e.target;const t=this.link.sourcePin??this.target,i=this.enteredPin,n=t.isOutput()?t:i;t.nodeElement.getType()===te.nodeType.knot||i.nodeElement.getType()===te.nodeType.knot?(this.link.setMessageCorrect(),this.linkValid=!0):t.getNodeElement()===i.getNodeElement()?this.link.setMessageSameNode():t.isOutput()===i.isOutput()?this.link.setMessageDirectionsIncompatible():this.blueprint.getLinks(t,i).length?(this.link.setMessageReplaceLink(),this.linkValid=!0):"exec"===n.entity.getType()&&n.isLinked?(this.link.setMessageReplaceOutputLink(),this.linkValid=!0):(this.link.setMessageCorrect(),this.linkValid=!0)}};#he=e=>{this.enteredPin==e.target&&(this.enteredPin=null,this.linkValid=!1,this.link?.setMessagePlaceNode())};link;enteredPin;linkValid=!1;constructor(e,t,i={}){i.scrollGraphEdge??=!0,super(e,t,i)}startDrag(e){this.target.nodeElement.getType()==te.nodeType.knot&&(this.#ue=this.target),this.link=rt.getConstructor("ueb-link").newObject(this.target,null),this.blueprint.template.linksContainerElement.prepend(this.link),this.link.setMessagePlaceNode(),this.#le=this.blueprint.querySelectorAll("ueb-pin"),this.#le.forEach((e=>{e!=this.target&&(e.addEventListener("mouseenter",this.#ce),e.addEventListener("mouseleave",this.#he))})),this.link.startDragging(),this.link.setDestinationLocation(e)}dragTo(e,t){this.link.setDestinationLocation(e)}endDrag(){if(this.#le.forEach((e=>{e.removeEventListener("mouseenter",this.#ce),e.removeEventListener("mouseleave",this.#he)})),this.enteredPin&&this.linkValid){if(this.#ue){const e=this.#ue!==this.link.sourcePin?this.link.sourcePin:this.enteredPin;if(this.#ue.isInput()&&e.isInput()||this.#ue.isOutput()&&e.isOutput()){const e=this.#ue.template.getOppositePin();this.#ue===this.link.sourcePin?this.link.sourcePin=e:this.enteredPin=e}}else this.enteredPin.nodeElement.getType()===te.nodeType.knot&&(this.enteredPin=this.enteredPin.template.getOppositePin());this.blueprint.addGraphElement(this.link),this.link.destinationPin=this.enteredPin,this.link.removeMessage(),this.link.finishDragging()}else this.link.finishDragging(),this.link.remove();this.enteredPin=null,this.link=null,this.#le=null}}class kt extends Ct{#de=!1;#pe=!1;#me="";static nodeStyleClasses=["ueb-node-style-glass"];initialize(e){super.initialize(e),this.#me=this.element.getNodeDisplayName()}render(){return $`
${this.#me?$`
${this.#me}
`:O}
${this.#de?$`
`:O} ${this.#pe?$`
`:O}
`}createPinElements(){return this.element.getPinEntities().filter((e=>!e.isHidden())).map((e=>{this.#de||=e.isInput(),this.#pe||=e.isOutput();return rt.getConstructor("ueb-pin").newObject(e,void 0,this.element)}))}}class Lt extends kt{static nodeStyleClasses=[...super.nodeStyleClasses,"ueb-node-style-conversion"]}class Nt extends kt{static nodeStyleClasses=[...super.nodeStyleClasses,"ueb-node-style-operation"]}class Tt extends Ke{static canWrapInput=!0;#ge;get iconElement(){return this.#ge}#be;get wrapperElement(){return this.#be}isNameRendered=!0;setup(){super.setup(),this.element.nodeElement=this.element.closest("ueb-node");const e=this.element.nodeElement.template;(e instanceof Lt||e instanceof Nt)&&(this.isNameRendered=!1,this.element.requestUpdate())}createInputObjects(){return[new Pt(this.element,this.blueprint,{moveEverywhere:!0,draggableElement:this.#be})]}render(){const e=$`
${this.renderIcon()}
`,t=$`
${this.isNameRendered?this.renderName():O} ${this.element.isInput()&&!this.element.entity.bDefaultValueIsIgnored?this.renderInput():$``}
`;return $`
${this.element.isInput()?$`${e}${t}`:$`${t}${e}`}
`}renderIcon(){switch(this.element.entity.PinType.ContainerType.toString()){case"Array":return $e.array;case"Set":return $e.set;case"Map":return $e.map}return"delegate"===this.element.entity.PinType.PinCategory?$e.delegate:$e.genericPin}renderName(){return $`${this.element.getPinDisplayName()}`}renderInput(){return $``}updated(e){if(super.updated(e),this.element.isInput()&&e.has("isLinked")){const e=this.element.nodeElement;e.addNextUpdatedCallbacks((()=>e.acknowledgeReflow())),e.requestUpdate()}}firstUpdated(e){super.firstUpdated(e),this.element.style.setProperty("--ueb-pin-color-rgb",this.element.entity.pinColor().cssText),this.#ge=this.element.querySelector(".ueb-pin-icon svg")??this.element,this.#be=this.element.querySelector(".ueb-pin-wrapper")}getLinkLocation(){const e=this.iconElement.getBoundingClientRect(),t=[this.element.isInput()?e.left:e.right,(e.top+e.bottom)/2],i=ae.convertLocation(t,this.blueprint.template.gridElement);return this.blueprint.compensateTranslation(i[0],i[1])}getClickableElement(){return this.#be??this.element}}class At extends Tt{render(){return $`
${this.renderIcon()}
`}}class Mt extends Ct{static nodeStyleClasses=[...super.nodeStyleClasses,"ueb-node-style-event"];firstUpdated(e){super.firstUpdated(e),this.element.querySelector(".ueb-node-top").appendChild(this.createDelegatePinElement())}renderTop(){const e=this.renderNodeIcon(),t=this.renderNodeName();return $`
${e?$`
${e}
`:O} ${t?$`
${t} ${this.hasSubtitle&&this.element.entity.FunctionReference.MemberParent?$`
Custom Event
`:O}
`:O}
`}createDelegatePinElement(){const e=rt.getConstructor("ueb-pin").newObject(this.element.getPinEntities().find((e=>!e.isHidden()&&"delegate"===e.PinType.PinCategory)),new At,this.element);return e.template.isNameRendered=!1,e}createPinElements(){return this.element.getPinEntities().filter((e=>!e.isHidden()&&"delegate"!==e.PinType.PinCategory)).map((e=>rt.getConstructor("ueb-pin").newObject(e,void 0,this.element)))}}class Bt extends tt{static properties={...super.properties,selected:{type:Boolean,attribute:"data-selected",reflect:!0,converter:ae.booleanConverter}};dragHandler=e=>this.addLocation(...e.detail.value);constructor(){super(),this.selected=!1,this.listeningDrag=!1}setup(){super.setup(),this.setSelected(this.selected)}cleanup(){super.cleanup(),this.blueprint.removeEventListener(te.nodeDragGeneralEventName,this.dragHandler)}setSelected(e=!0){this.selected=e,this.blueprint&&(this.selected?(this.listeningDrag=!0,this.blueprint.addEventListener(te.nodeDragGeneralEventName,this.dragHandler)):(this.blueprint.removeEventListener(te.nodeDragGeneralEventName,this.dragHandler),this.listeningDrag=!1))}}class Dt extends At{render(){return this.element.isOutput()?super.render():$``}getOppositePin(){const e=this.element.nodeElement.template;return this.element.isOutput()?e.inputPin:e.outputPin}getLinkLocation(){const e=(this.element.isInput()?this.element.nodeElement.template.outputPin.template:this).iconElement.getBoundingClientRect(),t=[this.element.isInput()?e.left:e.right,(e.top+e.bottom)/2],i=ae.convertLocation(t,this.blueprint.template.gridElement);return this.blueprint.compensateTranslation(i[0],i[1])}}class $t extends Ct{static#ve=new Set;#fe=null;#ye;get inputPin(){return this.#ye}#we;get outputPin(){return this.#we}initialize(e){super.initialize(e),this.element.classList.add("ueb-node-style-minimal")}findDirectionaPin(e){if(e.nodeElement.getType()!==te.nodeType.knot||$t.#ve.has(e))return $t.#ve.clear(),!0;$t.#ve.add(e);for(let t of e.getLinks().map((e=>this.blueprint.getPin(e))))if(this.findDirectionaPin(t))return!0;return!1}render(){return $`
`}setupPins(){this.element.getPinElements().forEach((e=>this.element.querySelector(".ueb-node-border").appendChild(e)))}getPinElements(e){return e.querySelectorAll("ueb-pin")}createPinElements(){const e=this.element.getPinEntities().filter((e=>!e.isHidden())),t=e[e[0].isInput()?0:1],i=e[e[0].isOutput()?0:1],n=rt.getConstructor("ueb-pin");return[this.#ye=n.newObject(t,new Dt,this.element),this.#we=n.newObject(i,new Dt,this.element)]}linksChanged(){}}class Ht extends kt{initialize(e){super.initialize(e),e.getType()===te.nodeType.variableGet?this.element.classList.add("ueb-node-style-getter"):e.getType()===te.nodeType.variableSet&&this.element.classList.add("ueb-node-style-setter")}setupPins(){super.setupPins();let e=this.element.getPinElements().find((e=>!e.entity.isHidden()&&!e.entity.isExecution()));this.element.style.setProperty("--ueb-node-color",e.getColor().cssText)}}class Ot extends Bt{static properties={...Bt.properties,typePath:{type:String,attribute:"data-type",reflect:!0},nodeName:{type:String,attribute:"data-name",reflect:!0},advancedPinDisplay:{type:String,attribute:"data-advanced-display",converter:fe.attributeConverter,reflect:!0},enabledState:{type:String,attribute:"data-enabled-state",reflect:!0},nodeDisplayName:{type:String,attribute:!1},pureFunction:{type:Boolean,converter:ae.booleanConverter,attribute:"data-pure-function",reflect:!0}};static dragEventName=te.nodeDragEventName;static dragGeneralEventName=te.nodeDragGeneralEventName;get blueprint(){return super.blueprint}set blueprint(e){super.blueprint=e,this.#Ee.forEach((t=>t.blueprint=e))}#Ce;get nodeNameElement(){return this.#Ce}set nodeNameElement(e){this.#Ce=e}#Ee=[];boundComments=[];#Se=!1;#xe=e=>{this.selected||this.#Se||(this.#Se=!0,this.addNextUpdatedCallbacks((()=>this.#Se=!1)),this.addLocation(...e.detail.value))};static getTypeTemplate(e){if(e.getClass()===te.nodeType.callFunction||e.getClass()===te.nodeType.commutativeAssociativeBinaryOperator||e.getClass()===te.nodeType.callArrayFunction){const t=e.FunctionReference.MemberParent?.path??"";if("/Script/Engine.KismetMathLibrary"===t||"/Script/Engine.KismetArrayLibrary"===t){if(e.FunctionReference.MemberName?.startsWith("Conv_"))return Lt;if(e.FunctionReference.MemberName?.startsWith("Percent_"))return Nt;switch(e.FunctionReference.MemberName){case"Array_Add":case"Array_Identical":case"Abs":case"Array_Add":case"BMax":case"BMin":case"Exp":case"FMax":case"FMin":case"Max":case"MaxInt64":case"Min":case"MinInt64":return Nt}}if("/Script/Engine.BlueprintSetLibrary"===t)return Nt;if("/Script/Engine.BlueprintMapLibrary"===t)return Nt}switch(e.getClass()){case te.nodeType.comment:return xt;case te.nodeType.event:case te.nodeType.customEvent:return Mt;case te.nodeType.promotableOperator:return Nt;case te.nodeType.knot:return $t;case te.nodeType.variableGet:case te.nodeType.variableSet:return Ht}return e.getDelegatePin()?Mt:Ct}static fromSerializedObject(e){e=e.trim();let t=ne.getSerializer(Oe).deserialize(e);return Ot.newObject(t)}static newObject(e=new Oe,t=new(Ot.getTypeTemplate(e))){const i=new Ot;return i.initialize(e,t),i}initialize(e=new Oe,t=new(Ot.getTypeTemplate(e))){super.initialize(e,t),this.#Ee=this.template.createPinElements(),this.typePath=this.entity.getType(),this.nodeName=this.entity.getObjectName(),this.advancedPinDisplay=this.entity.AdvancedPinDisplay?.toString(),this.enabledState=this.entity.EnabledState,this.nodeDisplayName=this.getNodeDisplayName(),this.pureFunction=this.entity.bIsPureFunc,this.dragLinkObjects=[],super.setLocation(this.entity.getNodePosX(),this.entity.getNodePosY()),this.entity.NodeWidth&&this.entity.NodeHeight?(this.sizeX=this.entity.NodeWidth.value,this.sizeY=this.entity.NodeHeight.value):this.updateComplete.then((()=>this.computeSizes()))}getUpdateComplete(){return Promise.all([super.getUpdateComplete(),...this.getPinElements().map((e=>e.updateComplete))]).then((()=>!0))}bindToComment(e){e==this||this.boundComments.includes(e)||(e.addEventListener(te.nodeDragEventName,this.#xe),this.boundComments.push(e))}unbindFromComment(e){const t=this.boundComments.indexOf(e);t>=0&&(e.removeEventListener(te.nodeDragEventName,this.#xe),this.boundComments[t]=this.boundComments[this.boundComments.length-1],this.boundComments.pop())}isInsideComment(e){return this.topBoundary()>=e.topBoundary()&&this.rightBoundary()<=e.rightBoundary()&&this.bottomBoundary()<=e.bottomBoundary()&&this.leftBoundary()>=e.leftBoundary()}getType(){return this.entity.getType()}getNodeName(){return this.entity.getObjectName()}getNodeDisplayName(){return this.entity.nodeDisplayName()}setNodeWidth(e){this.entity.setNodeWidth(e),this.sizeX=e,this.acknowledgeReflow()}setNodeHeight(e){this.entity.setNodeHeight(e),this.sizeY=e,this.acknowledgeReflow()}sanitizeLinks(e=[]){this.getPinElements().forEach((t=>t.sanitizeLinks(e)))}rename(e){if(this.entity.Name==e)return!1;for(let t of this.getPinElements())for(let i of t.getLinks())this.blueprint.getPin(i).redirectLink(t,new Pe({objectName:e,pinGuid:t.entity.PinId}));this.entity.Name=e,this.nodeName=this.entity.Name}getPinElements(){return this.#Ee}getPinEntities(){return this.entity.CustomProperties.filter((e=>e instanceof De))}setLocation(e=0,t=0,i=!0){this.entity.setNodePosX(e),this.entity.setNodePosY(t),super.setLocation(e,t,i)}acknowledgeReflow(){this.requestUpdate(),this.updateComplete.then((()=>this.computeSizes()));let e=new CustomEvent(te.nodeReflowEventName);this.dispatchEvent(e)}setShowAdvancedPinDisplay(e){this.entity.AdvancedPinDisplay=new fe(e?"Shown":"Hidden"),this.advancedPinDisplay=this.entity.AdvancedPinDisplay}toggleShowAdvancedPinDisplay(){this.setShowAdvancedPinDisplay("Shown"!=this.entity.AdvancedPinDisplay?.toString())}}class Vt extends et{static properties={selecting:{type:Boolean,attribute:"data-selecting",reflect:!0,converter:ae.booleanConverter},scrolling:{type:Boolean,attribute:"data-scrolling",reflect:!0,converter:ae.booleanConverter},focused:{type:Boolean,attribute:"data-focused",reflect:!0,converter:ae.booleanConverter},zoom:{type:Number,attribute:"data-zoom",reflect:!0},scrollX:{type:Number,attribute:!1},scrollY:{type:Number,attribute:!1},additionalX:{type:Number,attribute:!1},additionalY:{type:Number,attribute:!1},translateX:{type:Number,attribute:!1},translateY:{type:Number,attribute:!1}};static nodeBoundariesSupplier=e=>({primaryInf:e.leftBoundary(!0),primarySup:e.rightBoundary(!0),secondaryInf:e.topBoundary(!0),secondarySup:e.bottomBoundary(!0)});static nodeSelectToggleFunction=(e,t)=>{e.setSelected(t)};#Pe=new Map;nodes=[];links=[];mousePosition=[0,0];waitingExpandUpdate=!1;constructor(){super(),this.selecting=!1,this.scrolling=!1,this.focused=!1,this.zoom=0,this.scrollX=te.expandGridSize,this.scrollY=te.expandGridSize,this.translateX=te.expandGridSize,this.translateY=te.expandGridSize,super.initialize({},new ut)}initialize(){}getGridDOMElement(){return this.template.gridElement}getScroll(){return[this.scrollX,this.scrollY]}setScroll(e,t){this.scrollX=e,this.scrollY=t}scrollDelta(e=0,t=0,i=!1,n=te.smoothScrollTime){if(i){let i=[0,0];ae.animate(0,e,n,(e=>{this.scrollDelta(e-i[0],0,!1),i[0]=e})),ae.animate(0,t,n,(e=>{this.scrollDelta(0,e-i[1],!1),i[1]=e}))}else{const i=[2*te.expandGridSize,2*te.expandGridSize];let n=this.getScroll(),s=[n[0]+e,n[1]+t],r=[0,0];for(let e=0;e<2;++e)s[e]i[e]-te.gridExpandThreshold*te.expandGridSize&&(r[e]=1);0==r[0]&&0==r[1]||this.seamlessExpand(r[0],r[1]),n=this.getScroll(),s=[n[0]+e,n[1]+t],this.setScroll(s[0],s[1])}}scrollCenter(){const e=this.getScroll(),t=[this.translateX-e[0],this.translateY-e[1]],i=this.getViewportSize().map((e=>e/2)),n=[t[0]-i[0],t[1]-i[1]];this.scrollDelta(n[0],n[1],!0)}getViewportSize(){return[this.template.viewportElement.clientWidth,this.template.viewportElement.clientHeight]}getScrollMax(){return[this.template.viewportElement.scrollWidth-this.template.viewportElement.clientWidth,this.template.viewportElement.scrollHeight-this.template.viewportElement.clientHeight]}snapToGrid(e,t){return ae.snapToGrid(e,t,te.gridSize)}seamlessExpand(e,t){e=Math.round(e),t=Math.round(t);let i=this.getScale();[e,t]=[-e*te.expandGridSize,-t*te.expandGridSize],0!=e&&(this.scrollX+=e,e/=i),0!=t&&(this.scrollY+=t,t/=i),this.translateX+=e,this.translateY+=t}progressiveSnapToGrid(e){return te.expandGridSize*Math.round(e/te.expandGridSize+.5*Math.sign(e))}getZoom(){return this.zoom}setZoom(e,t){if((e=ae.clamp(e,te.minZoom,te.maxZoom))==this.zoom)return;let i=this.getScale();if(this.zoom=e,t){t[0]+=this.translateX,t[1]+=this.translateY;let e=this.getScale()/i,n=[e*t[0],e*t[1]];this.scrollDelta((n[0]-t[0])*i,(n[1]-t[1])*i)}}getScale(){return te.scale[this.getZoom()]}scaleCorrect(e){return e/this.getScale()}scaleCorrectReverse(e){return e*this.getScale()}compensateTranslation(e,t){return[e-=this.translateX,t-=this.translateY]}getNodes(e=!1,[t,i,n,s]=[Number.MIN_SAFE_INTEGER,Number.MAX_SAFE_INTEGER,Number.MAX_SAFE_INTEGER,Number.MIN_SAFE_INTEGER]){let r=this.nodes;return e&&(r=r.filter((e=>e.selected))),(t>Number.MIN_SAFE_INTEGER||iNumber.MIN_SAFE_INTEGER)&&(r=r.filter((e=>e.topBoundary()>=t&&e.rightBoundary()<=i&&e.bottomBoundary()<=n&&e.leftBoundary()>=s))),r}getCommentNodes(e=!1){let t=[...this.template.getCommentNodes(e)];return 0===t.length&&(t=this.nodes.filter((t=>t.getType()===te.nodeType.comment&&(!e||t.selected)))),t}getPin(e){let t=this.template.getPin(e);return t&&t.nodeElement.getNodeName()==e.objectName.toString()||(t=[...this.nodes.find((t=>e.objectName.toString()==t.getNodeName()))?.getPinElements()??[]].find((t=>e.pinGuid.toString()==t.getPinId().toString()))),t}getLinks(e=null,t=null){if(null==e!=(null==t)){const i=e??t;return this.links.filter((e=>e.sourcePin==i||e.destinationPin==i))}return null!=e&&null!=t?this.links.filter((i=>i.sourcePin==e&&i.destinationPin==t||i.sourcePin==t&&i.destinationPin==e)):this.links}getLink(e,t,i=!1){return this.links.find((n=>n.sourcePin==e&&n.destinationPin==t||!i&&n.sourcePin==t&&n.destinationPin==e))}selectAll(){this.getNodes().forEach((e=>Vt.nodeSelectToggleFunction(e,!0)))}unselectAll(){this.getNodes().forEach((e=>Vt.nodeSelectToggleFunction(e,!1)))}addGraphElement(...e){const t=e=>{const i=e.currentTarget;i.removeEventListener(te.removeEventName,t);const n=i instanceof Ot?this.nodes:i instanceof gt?this.links:null,s=n?.indexOf(i);if(s>=0){const e=n.pop();st.entity.getObjectName()==e));if(n){let e=n.entity.getObjectName(!0);this.#Pe[e]=this.#Pe[e]??-1;do{++this.#Pe[e]}while(this.nodes.find((t=>t.entity.getObjectName()==te.nodeName(e,this.#Pe[e]))));n.rename(te.nodeName(e,this.#Pe[e]))}this.nodes.push(i),i.addEventListener(te.removeEventName,t),this.template.nodesContainerElement?.appendChild(i)}else i instanceof gt&&!this.links.includes(i)&&(this.links.push(i),i.addEventListener(te.removeEventName,t),this.template.linksContainerElement&&!this.template.linksContainerElement.contains(i)&&this.template.linksContainerElement.appendChild(i));e.filter((e=>e instanceof Ot)).forEach((t=>t.sanitizeLinks(e))),e.filter((e=>e instanceof Ot&&e.getType()==te.nodeType.comment)).forEach((e=>e.updateComplete.then((()=>e.template.manageNodesBind()))))}removeGraphElement(...e){for(let t of e){if(t.closest("ueb-blueprint")!==this)return;t.remove()}}setFocused(e=!0){if(this.focused==e)return;let t=new CustomEvent(e?te.focusEventName.begin:te.focusEventName.end);this.focused=e,this.focused||this.unselectAll(),this.dispatchEvent(t)}acknowledgeEditText(e){const t=new CustomEvent(e?te.editTextEventName.begin:te.editTextEventName.end);this.dispatchEvent(t)}}customElements.define("ueb-blueprint",Vt);class zt extends ft{#ke;get locationChangeCallback(){return this.#ke}set locationChangeCallback(e){this.#ke=e}movementSpace;movementSpaceSize=[0,0];firstUpdated(e){super.firstUpdated(e),this.movementSpace=this.element.parentElement}setup(){super.setup();const e=this.movementSpace.getBoundingClientRect();this.movementSpaceSize=[e.width,e.height]}createDraggableObject(){return new bt(this.element,this.blueprint,{draggableElement:this.movementSpace,ignoreTranslateCompensate:!0,moveEverywhere:!0,movementSpace:this.movementSpace,repositionOnClick:!0,stepSize:1})}adjustLocation(e,t){return this.locationChangeCallback?.(e,t),[e,t]}}class It extends zt{adjustLocation(e,t){const i=Math.round(this.movementSpaceSize[0]/2);e-=i,t=-(t-i);let[n,s]=ae.getPolarCoordinates(e,t);return n=Math.min(n,i),[e,t]=ae.getCartesianCoordinates(n,s),this.locationChangeCallback?.(e/i,t/i),[e=Math.round(e+i),t=Math.round(-t+i)]}}class Rt extends tt{windowElement;setup(){super.setup(),this.windowElement=this.closest("ueb-window")}setLocation(e,t){super.setLocation(...this.template.adjustLocation(e,t))}}class _t extends Rt{constructor(){super(),super.initialize({},new It)}static newObject(){return new _t}initialize(){}}class Ft extends zt{adjustLocation(e,t){return e=ae.clamp(e,0,this.movementSpaceSize[0]),t=ae.clamp(t,0,this.movementSpaceSize[1]),this.locationChangeCallback?.(e/this.movementSpaceSize[0],1-t/this.movementSpaceSize[1]),[e,t]}}class Gt extends Rt{constructor(){super(),super.initialize({},new Ft)}static newObject(){return new Gt}initialize(){}}class jt extends Ke{#Le=()=>{this.blueprint.acknowledgeEditText(!0),this.element.selectOnFocus&&getSelection().selectAllChildren(this.element)};#Ne=()=>{this.blueprint.acknowledgeEditText(!1),getSelection().removeAllRanges()};#Te=e=>e.target.querySelectorAll("br").forEach((e=>e.remove()));#Ae=e=>{"Enter"!=e.code||e.shiftKey||e.target.blur()};initialize(e){super.initialize(e),this.element.classList.add("ueb-pin-input-content"),this.element.setAttribute("role","textbox"),this.element.contentEditable="true"}setup(){super.setup(),this.element.addEventListener("focus",this.#Le),this.element.addEventListener("focusout",this.#Ne),this.element.singleLine&&this.element.addEventListener("input",this.#Te),this.element.blurOnEnter&&this.element.addEventListener("keydown",this.#Ae)}cleanup(){super.cleanup(),this.element.removeEventListener("focus",this.#Le),this.element.removeEventListener("focusout",this.#Ne),this.element.removeEventListener("input",this.#Te),this.element.removeEventListener("keydown",this.#Ae)}}class Ut extends et{static properties={...super.properties,singleLine:{type:Boolean,attribute:"data-single-line",converter:ae.booleanConverter,reflect:!0},selectOnFocus:{type:Boolean,attribute:"data-select-focus",converter:ae.booleanConverter,reflect:!0},blurOnEnter:{type:Boolean,attribute:"data-blur-enter",converter:ae.booleanConverter,reflect:!0}};constructor(){super(),this.singleLine=!1,this.selectOnFocus=!0,this.blurOnEnter=!0,super.initialize({},new jt)}static newObject(){return new Ut}initialize(){}}class Kt extends it{constructor(e,t,i={}){i.consumeEvent=!0,super(e,t,i)}}class Wt extends Tt{#Me;#Be=e=>this.element.setDefaultValue(this.#Me.checked);firstUpdated(e){super.firstUpdated(e),this.#Me=this.element.querySelector(".ueb-pin-input")}setup(){super.setup(),this.#Me?.addEventListener("change",this.#Be)}cleanup(){super.cleanup(),this.#Me?.removeEventListener("change",this.#Be)}createInputObjects(){return[...super.createInputObjects(),new Kt(this.#Me,this.blueprint)]}renderInput(){return $``}}class Zt extends Tt{renderIcon(){return $e.execPin}renderName(){let e=this.element.entity.PinName;if(this.element.entity.PinFriendlyName)e=this.element.entity.PinFriendlyName.toString();else if("execute"===e||"then"===e)return $``;return $`${ae.formatStringName(e)}`}}class Xt extends Tt{static singleLineInput=!1;static selectOnFocus=!0;#De;get inputContentElements(){return this.#De}static stringFromInputToUE(e){return e.replace(/(?=\n\s*)\n$/,"").replaceAll("\n","\\r\n")}static stringFromUEToInput(e){return e.replaceAll(/(?:\r|(?<=(?:^|[^\\])(?:\\\\)*)\\r)(?=\n)/g,"").replace(/(?<=\n\s*)$/,"\n")}#$e=()=>this.setInputs(this.getInputs(),!0);#He=e=>this.#Oe(e.target);#Oe(e){const t=this.blueprint.scaleCorrect(e.getBoundingClientRect().width)+this.nameWidth,i=this.element.classList.contains("ueb-pin-input-wrap");!i&&t>te.pinInputWrapWidth?this.element.classList.add("ueb-pin-input-wrap"):i&&t<=te.pinInputWrapWidth&&this.element.classList.remove("ueb-pin-input-wrap")}firstUpdated(e){super.firstUpdated(e),this.#De=[...this.element.querySelectorAll("ueb-input")],this.constructor.canWrapInput&&(this.nameWidth=this.blueprint.scaleCorrect(this.element.querySelector(".ueb-pin-name").getBoundingClientRect().width),this.inputContentElements.forEach((e=>this.#Oe(e))))}setup(){super.setup(),this.#De.forEach((e=>{e.addEventListener("focusout",this.#$e),this.constructor.canWrapInput&&e.addEventListener("input",this.#He)}))}cleanup(){super.cleanup(),this.#De.forEach((e=>{e.removeEventListener("focusout",this.#$e),e.removeEventListener("input",this.#He)}))}createInputObjects(){return[...super.createInputObjects(),...this.#De.map((e=>new Kt(e,this.blueprint)))]}getInput(){return this.getInputs().reduce(((e,t)=>e+t),"")}getInputs(){return this.#De.map((e=>ae.clearHTMLWhitespace(e.innerHTML)))}setInputs(e=[],t=!0){this.#De.forEach(this.constructor.singleLineInput?(t,i)=>t.innerText=e[i]:(t,i)=>t.innerText=e[i].replaceAll("\n","")),t&&this.setDefaultValue(e.map((e=>Xt.stringFromInputToUE(e))),e),this.element.addNextUpdatedCallbacks((()=>this.element.nodeElement.acknowledgeReflow()))}setDefaultValue(e=[],t=e){this.element.setDefaultValue(e.join(""))}renderInput(){const e=this.constructor.singleLineInput,t=this.constructor.selectOnFocus;return $`
`}}class Yt extends Xt{static singleLineInput=!0;setInputs(e=[],t=!1){if(e&&0!=e.length||(e=[this.getInput()]),super.setInputs(e,!1),t){let i=[];for(const n of e){let e=parseFloat(n);isNaN(e)&&(e=0,t=!1),i.push(e)}this.setDefaultValue(i,e)}}setDefaultValue(e=[],t){this.element.setDefaultValue(e[0])}}class qt extends Yt{setDefaultValue(e=[],t=e){this.element.getDefaultValue(!0).value=e[0],this.inputContentElements[0].innerText=this.element.getDefaultValue()?.toString(),this.element.requestUpdate()}renderInput(){return $`
`}}class Jt extends qt{setInputs(e=[],t=!1){if(e&&0!=e.length||(e=[this.getInput()]),super.setInputs(e,!1),t){if(!e[0].match(/[\-\+]?[0-9]+/))return;const t=[BigInt(e[0])];this.setDefaultValue(t,e)}}} /** * @license * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause - */const Jt=1;class Qt{constructor(e){}get _$AU(){return this._$AM._$AU}_$AT(e,t,i){this._$Ct=e,this._$AM=t,this._$Ci=i}_$AS(e,t){return this.update(e,t)}update(e,t){return this.render(...t)}} + */const Qt=1;class ei{constructor(e){}get _$AU(){return this._$AM._$AU}_$AT(e,t,i){this._$Ct=e,this._$AM=t,this._$Ci=i}_$AS(e,t){return this.update(e,t)}update(e,t){return this.render(...t)}} /** * @license * Copyright 2018 Google LLC * SPDX-License-Identifier: BSD-3-Clause - */const ei=(e=>(...t)=>({_$litDirective$:e,values:t}))(class extends Qt{constructor(e){var t;if(super(e),e.type!==Jt||"style"!==e.name||(null===(t=e.strings)||void 0===t?void 0:t.length)>2)throw Error("The `styleMap` directive must be used in the `style` attribute and must be the only part in the attribute.")}render(e){return Object.keys(e).reduce(((t,i)=>{const n=e[i];return null==n?t:t+`${i=i.replace(/(?:^(webkit|moz|ms|o)|)(?=[A-Z])/g,"-$&").toLowerCase()}:${n};`}),"")}update(e,[t]){const{style:i}=e.element;if(void 0===this.vt){this.vt=new Set;for(const e in t)this.vt.add(e);return this.render(t)}this.vt.forEach((e=>{null==t[e]&&(this.vt.delete(e),e.includes("-")?i.removeProperty(e):i[e]="")}));for(const e in t){const n=t[e];null!=n&&(this.vt.add(e),e.includes("-")?i.setProperty(e,n):i[e]=n)}return H}});class ti extends ft{toggleAdvancedDisplayHandler;getDraggableElement(){return this.element.querySelector(".ueb-window-top")}createDraggableObject(){return new gt(this.element,this.blueprint,{draggableElement:this.getDraggableElement(),ignoreScale:!0,ignoreTranslateCompensate:!1,movementSpace:this.blueprint,stepSize:1})}setup(){const e=this.blueprint.template.gridLeftVisibilityBoundary(),t=this.blueprint.template.gridTopVisibilityBoundary();this.element.locationX=this.blueprint.scaleCorrectReverse(this.blueprint.mousePosition[0]-e),this.element.locationY=this.blueprint.scaleCorrectReverse(this.blueprint.mousePosition[1]-t),this.element.updateComplete.then((()=>{const e=this.blueprint.getBoundingClientRect();this.element.locationX+this.element.sizeX>e.width&&(this.element.locationX=e.width-this.element.sizeX),this.element.locationX=Math.max(0,this.element.locationX),this.element.locationY+this.element.sizeY>e.height&&(this.element.locationY=e.height-this.element.sizeY),this.element.locationY=Math.max(0,this.element.locationY)}))}render(){return $`
${this.renderWindowName()}
${De.close}
${this.renderContent()}
`}renderWindowName(){return $`Window`}renderContent(){return $``}apply(){this.element.dispatchEvent(new CustomEvent(te.windowApplyEventName)),this.element.remove()}cancel(){this.element.dispatchEvent(new CustomEvent(te.windowCancelEventName)),this.element.remove()}}class ii extends ti{#Ve;#ze;#Ie;#Re;#_e;#Fe;#Ge;#je;#Ue;#Ke;#We=e=>{const t=ae.clearHTMLWhitespace(e.target.innerHTML),i=parseInt(t,16);isNaN(i)||(this.color.setFromRGBANumber(i),this.element.requestUpdate())};#Xe=e=>{const t=ae.clearHTMLWhitespace(e.target.innerHTML),i=parseInt(t,16);isNaN(i)||(this.color.setFromSRGBANumber(i),this.element.requestUpdate())};#Ze=e=>t=>{"Enter"==t.code&&(t.preventDefault(),e(t))};#ae=new we;get color(){return this.#ae}set color(e){e.toNumber()!=this.color?.toNumber()&&(this.element.requestUpdate("color",this.#ae),this.#ae=e)}#Ye=new we;get fullColor(){return this.#Ye}#qe;get initialColor(){return this.#qe}#Je=new we;#Qe(e,t,i=!1){const n=this.color.toRGBAString(),s=`${n.substring(0,2*e)}${t}${n.substring(2+2*e)}`;return i?`${s.substring(0,6)}FF`:s}initialize(e){super.initialize(e),this.#qe=this.element.windowOptions.getPinColor(),this.color.setFromHSVA(this.initialColor.H.value,this.initialColor.S.value,this.initialColor.V.value,this.initialColor.A.value),this.fullColor.setFromHSVA(this.color.H.value,1,1,1)}firstUpdated(e){this.#Ve=this.element.querySelector(".ueb-color-picker-wheel ueb-color-handler"),this.#ze=this.element.querySelector(".ueb-color-picker-saturation ueb-ui-slider"),this.#Ie=this.element.querySelector(".ueb-color-picker-value ueb-ui-slider"),this.#Re=this.element.querySelector(".ueb-color-picker-r ueb-ui-slider"),this.#_e=this.element.querySelector(".ueb-color-picker-g ueb-ui-slider"),this.#Fe=this.element.querySelector(".ueb-color-picker-b ueb-ui-slider"),this.#Ge=this.element.querySelector(".ueb-color-picker-a ueb-ui-slider"),this.#je=this.element.querySelector(".ueb-color-picker-h ueb-ui-slider"),this.#Ue=this.element.querySelector(".ueb-color-picker-s ueb-ui-slider"),this.#Ke=this.element.querySelector(".ueb-color-picker-v ueb-ui-slider"),this.#Ve.template.locationChangeCallback=(e,t)=>{this.color.setFromWheelLocation(e,t,this.color.V.value,this.color.A.value),this.fullColor.setFromHSVA(this.color.H.value,1,1,1),this.element.requestUpdate()},this.#ze.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,t,this.color.V.value,this.color.A.value),this.element.requestUpdate()},this.#Ie.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,this.color.S.value,t,this.color.A.value),this.element.requestUpdate()},this.#Re.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(e,this.color.G.value,this.color.B.value,this.color.A.value),this.element.requestUpdate()},this.#_e.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(this.color.R.value,e,this.color.B.value,this.color.A.value),this.element.requestUpdate()},this.#Fe.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(this.color.R.value,this.color.G.value,e,this.color.A.value),this.element.requestUpdate()},this.#Ge.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(this.color.R.value,this.color.G.value,this.color.B.value,e),this.element.requestUpdate()},this.#je.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(e,this.color.S.value,this.color.V.value,this.color.A.value),this.element.requestUpdate()},this.#Ue.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,e,this.color.V.value,this.color.A.value),this.element.requestUpdate()},this.#Ke.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,this.color.S.value,e,this.color.A.value),this.element.requestUpdate()}}renderSlider(e){let t="",i=0,n="";const s=e=>`linear-gradient(to right, #${this.#Qe(e,"00",!0)}, #${this.#Qe(e,"ff",!0)})`;switch(e){case 0:t="r",i=this.color.R.value,n=s(e);break;case 1:t="g",i=this.color.G.value,n=s(e);break;case 2:t="b",i=this.color.B.value,n=s(e);break;case 3:t="a",i=this.color.A.value,n=`${te.alphaPattern}, ${s(e)}`;break;case 4:t="h",i=360*this.color.H.value,n="linear-gradient(to right, #f00 0%, #ff0 16.666%, #0f0 33.333%, #0ff 50%, #00f 66.666%, #f0f 83.333%, #f00 100%)";break;case 5:t="s",i=this.color.S.value,n=`linear-gradient(to right,#${this.#Je.setFromHSVA(this.color.H.value,0,this.color.V.value,1),this.#Je.toRGBAString()},#${this.#Je.setFromHSVA(this.color.H.value,1,this.color.V.value,1),this.#Je.toRGBAString()})`;break;case 6:t="v",i=this.color.V.value,n=`linear-gradient(to right, #000, #${this.fullColor.toRGBAString()})`}return n=`background: ${n};`,$`
${t.toUpperCase()}
`}renderContent(){const e=2*this.color.H.value*Math.PI,t={"--ueb-color-r":this.color.R.toString(),"--ueb-color-g":this.color.G.toString(),"--ueb-color-b":this.color.B.toString(),"--ueb-color-a":this.color.A.toString(),"--ueb-color-h":this.color.H.toString(),"--ueb-color-s":this.color.S.toString(),"--ueb-color-v":this.color.V.toString(),"--ueb-color-wheel-x":100*(this.color.S.value*Math.cos(e)*.5+.5)+"%","--ueb-color-wheel-y":100*(this.color.S.value*Math.sin(e)*.5+.5)+"%"},i=this.color.toRGBAString(),n=this.color.toSRGBAString(),s=this.fullColor.toRGBAString();return $`
Old
New
Advanced
${this.renderSlider(0)} ${this.renderSlider(1)} ${this.renderSlider(2)} ${this.renderSlider(3)}
${this.renderSlider(4)} ${this.renderSlider(5)} ${this.renderSlider(6)}
Hex Linear
Hex sRGB
${te.windowApplyButtonText}
${te.windowCancelButtonText}
`}renderWindowName(){return $`${te.colorWindowName}`}}class ni extends Nt{#et;#tt=e=>{e.preventDefault(),this.blueprint.setFocused(!0),this.#et=st.getConstructor("ueb-window").newObject({type:new ii,windowOptions:{getPinColor:()=>this.element.defaultValue,setPinColor:e=>this.element.setDefaultValue(e)}}),this.blueprint.append(this.#et);const t=()=>{this.element.setDefaultValue(this.#et.template.color)},i=()=>{this.#et.removeEventListener(te.windowApplyEventName,t),this.#et.removeEventListener(te.windowCloseEventName,i),this.#et=null};this.#et.addEventListener(te.windowApplyEventName,t),this.#et.addEventListener(te.windowCloseEventName,i)};renderInput(){return $``}}class si extends Xt{static singleLineInput=!0}class ri extends Zt{setDefaultValue(e=[],t=e){this.element.setDefaultValue(e[0])}renderInput(){return $`
`}}class ai extends Nt{renderIcon(){return De.referencePin}}class oi extends Zt{#it(){return ae.minDecimals(this.element.getDefaultValue()?.R??0)}#nt(){return ae.minDecimals(this.element.getDefaultValue()?.P??0)}#st(){return ae.minDecimals(this.element.getDefaultValue()?.Y??0)}setDefaultValue(e=[],t=e){const i=this.element.getDefaultValue(!0);if(!(i instanceof ke))throw new TypeError("Expected DefaultValue to be a RotatorEntity");i.R=e[0],i.P=e[1],i.Y=e[2],this.element.requestUpdate("DefaultValue",i)}renderInput(){return $`
X
Y
Z
`}}class li extends Xt{}class ui extends Zt{#rt(){return ae.minDecimals(this.element.getDefaultValue()?.X??0)}#st(){return ae.minDecimals(this.element.getDefaultValue()?.Y??0)}setDefaultValue(e,t){const i=this.element.getDefaultValue(!0);if(!(i instanceof Ne))throw new TypeError("Expected DefaultValue to be a Vector2DEntity");i.X=e[0],i.Y=e[1],this.element.requestUpdate("DefaultValue",i)}renderInput(){return $`
X
Y
`}}class ci extends Zt{#rt(){return ae.minDecimals(this.element.getDefaultValue()?.X??0)}#st(){return ae.minDecimals(this.element.getDefaultValue()?.Y??0)}#at(){return ae.minDecimals(this.element.getDefaultValue()?.Z??0)}setDefaultValue(e,t){const i=this.element.getDefaultValue(!0);if(!(i instanceof Ae))throw new TypeError("Expected DefaultValue to be a VectorEntity");i.X=e[0],i.Y=e[1],i.Z=e[2],this.element.requestUpdate("DefaultValue",i)}renderInput(){return $`
X
Y
Z
`}}class hi extends Qe{static#ot={"/Script/CoreUObject.LinearColor":ni,"/Script/CoreUObject.Rotator":oi,"/Script/CoreUObject.Vector":ci,"/Script/CoreUObject.Vector2D":ui,bool:Kt,byte:Yt,int:Yt,int64:qt,MUTABLE_REFERENCE:ai,name:si,real:ri,string:li};static properties={pinId:{type:de,converter:{fromAttribute:(e,t)=>e?Fe.grammar.Guid.parse(e).value:null,toAttribute:(e,t)=>e?.toString()},attribute:"data-id",reflect:!0},pinType:{type:String,attribute:"data-type",reflect:!0},advancedView:{type:String,attribute:"data-advanced-view",reflect:!0},color:{type:we,converter:{fromAttribute:(e,t)=>e?Fe.grammar.LinearColorFromAnyColor.parse(e).value:null,toAttribute:(e,t)=>e?ae.printLinearColor(e):null},attribute:"data-color",reflect:!0},defaultValue:{type:String,attribute:!1},isLinked:{type:Boolean,converter:ae.booleanConverter,attribute:"data-linked",reflect:!0},pinDirection:{type:String,attribute:"data-direction",reflect:!0}};nodeElement;static getTypeTemplate(e){if(e.PinType.bIsReference&&!e.PinType.bIsConst)return hi.#ot.MUTABLE_REFERENCE;if("exec"===e.getType())return Wt;let t;return e.isInput()&&(t=hi.#ot[e.getType()]),t??Nt}static newObject(e=new Be,t=new(hi.getTypeTemplate(e)),i){const n=new hi;return n.initialize(e,t,i),n}initialize(e=new Be,t=new(hi.getTypeTemplate(e)),i){super.initialize(e,t),this.pinId=this.entity.PinId,this.pinType=this.entity.getType(),this.advancedView=this.entity.bAdvancedView,this.defaultValue=this.entity.getDefaultValue(),this.color=hi.properties.color.converter.fromAttribute(this.getColor().toString()),this.isLinked=!1,this.pinDirection=e.isInput()?"input":e.isOutput()?"output":"hidden",this.nodeElement=i}setup(){super.setup(),this.nodeElement=this.closest("ueb-node")}createPinReference(){return new Pe({objectName:this.nodeElement.getNodeName(),pinGuid:this.getPinId()})}getPinId(){return this.entity.PinId}getPinName(){return this.entity.PinName}getPinDisplayName(){return this.entity.getDisplayName()}getColor(){return this.entity.pinColor()}isInput(){return this.entity.isInput()}isOutput(){return this.entity.isOutput()}getLinkLocation(){return this.template.getLinkLocation()}getNodeElement(){return this.nodeElement}getLinks(){return this.entity.LinkedTo??[]}getDefaultValue(e=!1){return this.defaultValue=this.entity.getDefaultValue(e)}setDefaultValue(e){this.entity.DefaultValue=e,this.defaultValue=e}sanitizeLinks(e=[]){this.entity.LinkedTo=this.entity.LinkedTo?.filter((t=>{let i=this.blueprint.getPin(t);if(i){if(e.length&&!e.includes(i.nodeElement))return!1;let t=this.blueprint.getLink(this,i);t||(t=st.getConstructor("ueb-link").newObject(this,i),this.blueprint.addGraphElement(t))}return i})),this.isLinked=this.entity.isLinked()}linkTo(e){const t=this.createPinReference();!this.isLinked||!this.isOutput()||"exec"!==this.pinType&&"exec"!==e.pinType||this.getLinks().some((e=>t.equals(e)))||this.unlinkFromAll(),this.entity.linkTo(e.getNodeElement().getNodeName(),e.entity)&&(this.isLinked=this.entity.isLinked(),this.nodeElement?.template.linksChanged())}unlinkFrom(e,t=!0){this.entity.unlinkFrom(e.getNodeElement().getNodeName(),e.entity)&&(this.isLinked=this.entity.isLinked(),this.nodeElement?.template.linksChanged(),t&&this.blueprint.getLink(this,e)?.remove())}unlinkFromAll(){const e=this.getLinks().length;this.getLinks().map((e=>this.blueprint.getPin(e))).forEach((e=>this.unlinkFrom(e))),e&&this.nodeElement?.template.linksChanged()}redirectLink(e,t){const i=this.getLinks().findIndex((t=>t.objectName.toString()==e.getNodeElement().getNodeName()&&t.pinGuid.valueOf()==e.entity.PinId.valueOf()));return i>=0&&(this.entity.LinkedTo[i]=t,!0)}}class di{constructor(e=(e=>e),t=null){this.array=new Uint32Array(t),this.comparisonValueSupplier=e,this.length=0,this.currentPosition=0}get(e){return e>=0&&e=0&&this.currentPosition=0&&this.currentPosition0?this.get(this.currentPosition-1):null}getPrevValue(){return this.currentPosition>0?this.comparisonValueSupplier(this.get(this.currentPosition-1)):Number.MIN_SAFE_INTEGER}shiftLeft(e,t=1){this.array.set(this.array.subarray(e+t),e)}shiftRight(e,t=1){this.array.set(this.array.subarray(e,-t),e+t)}}class pi{constructor(e,t,i,n){this.initialPosition=e,this.finalPosition=e,this.metadata=new Array(t.length),this.primaryOrder=new di((e=>this.metadata[e].primaryBoundary)),this.secondaryOrder=new di((e=>this.metadata[e].secondaryBoundary)),this.selectFunc=n,this.rectangles=t,this.primaryOrder.reserve(this.rectangles.length),this.secondaryOrder.reserve(this.rectangles.length),t.forEach(((e,t)=>{let s={primaryBoundary:this.initialPosition[0],secondaryBoundary:this.initialPosition[1],rectangle:t,onSecondaryAxis:!1};this.metadata[t]=s,n(e,!1);const r=i(e);this.initialPosition[1]{if(this.metadata[i].onSecondaryAxis)this.selectFunc(this.rectangles[i],n);else if(n){this.secondaryOrder.insert(i,e[1]);const n=this.metadata[i].secondaryBoundary;Math.sign(e[1]-n)==t[1]&&Math.sign(n-this.initialPosition[1])==t[1]&&this.selectFunc(this.rectangles[i],!0)}else this.selectFunc(this.rectangles[i],!1),this.secondaryOrder.remove(i);this.computeBoundaries(),this.selectTo(e)};e[0]this.boundaries.primaryN.v&&e[0]this.boundaries.primaryP.v&&(++this.primaryOrder.currentPosition,i(this.boundaries.primaryP.i,this.initialPosition[0]{this.selectFunc(this.rectangles[t],i),this.computeBoundaries(),this.selectTo(e)};e[1]this.boundaries.secondaryN.v&&e[1]this.boundaries.secondaryP.v&&(++this.secondaryOrder.currentPosition,n(this.boundaries.secondaryP.i,this.initialPosition[1]bi.#lt[e],toAttribute:(e,t)=>Object.entries(bi.#lt).find((([t,i])=>e.constructor===i))?.[0]}}};static newObject(e={},t=e.type??new ti){const i=new bi;return i.initialize(e,t),i}initialize(e={},t=e.type??new ti){e.windowOptions??={},this.type=e.type,this.windowOptions=e.windowOptions,super.initialize(e,t)}computeSizes(){const e=this.getBoundingClientRect();this.sizeX=e.width,this.sizeY=e.height}cleanup(){super.cleanup(),this.acknowledgeClose()}acknowledgeClose(){let e=new CustomEvent(te.windowCloseEventName);this.dispatchEvent(e)}}class vi extends Fe{constructor(e,t,i,n,s,r,a){e=e??(e=>`(${e})`),super(t,i,n,s,r,a),this.wrap=e}read(e){const t=_e.getGrammarForType(Fe.grammar,this.entityType).parse(e);if(!t.status)throw new Error(`Error when trying to parse the entity ${this.entityType.prototype.constructor.name}.`);return t.value}write(e,t,i=!1){return this.wrap(this.subWrite(e,[],t,i),t)}}class fi extends vi{#ut;constructor(e,t){super(void 0,t),this.#ut=e}write(e,t,i=!1){return this.#ut(t,i)}}class yi extends vi{constructor(e){super(void 0,e)}write(e,t,i){return i||t.constructor!==String?ae.escapeString(t.toString()):`"${ae.escapeString(t.toString())}"`}}!function(){const e=e=>`(${e})`;ne.registerSerializer(null,new fi(((e,t)=>"()"),null)),ne.registerSerializer(Array,new fi(((e,t)=>`(${e.map((e=>ne.getSerializer(ae.getType(e)).serialize(e,t)+",")).join("")})`),Array)),ne.registerSerializer(BigInt,new yi(BigInt)),ne.registerSerializer(Boolean,new fi(((e,t)=>e?t?"true":"True":t?"false":"False"),Boolean)),ne.registerSerializer(ue,new yi(ue)),ne.registerSerializer(he,new yi(he)),ne.registerSerializer(me,new vi(e,me)),ne.registerSerializer(de,new yi(de)),ne.registerSerializer(ge,new yi(ge)),ne.registerSerializer(be,new yi(be)),ne.registerSerializer(le,new yi(le)),ne.registerSerializer(ve,new vi((e=>`${ve.lookbehind}(${e})`),ve,"",", ",!1,"",(e=>""))),ne.registerSerializer(fe,new vi(e,fe)),ne.registerSerializer(we,new vi(e,we)),ne.registerSerializer(Ee,new vi((e=>`${Ee.lookbehind}(${e})`),Ee,"",", ",!1,"",(e=>""))),ne.registerSerializer(Ce,new vi(e,Ce)),ne.registerSerializer(Number,new yi(Number)),ne.registerSerializer(He,new Ge),ne.registerSerializer(pe,new fi((e=>(e.type??"")+(e.path?e.type?`'"${e.path}"'`:`"${e.path}"`:"")),pe)),ne.registerSerializer(Se,new yi(Se)),ne.registerSerializer(Be,new vi((e=>`${Be.lookbehind} (${e})`),Be,"",",",!0)),ne.registerSerializer(Pe,new vi((e=>e),Pe,""," ",!1,"",(e=>""))),ne.registerSerializer(xe,new vi(e,xe)),ne.registerSerializer(ye,new yi(ye)),ne.registerSerializer(ke,new vi(e,ke)),ne.registerSerializer(String,new fi(((e,t)=>t?ae.escapeString(e):`"${ae.escapeString(e)}"`),String)),ne.registerSerializer(Le,new fi(((e,t)=>`${e.P}, ${e.Y}, ${e.R}`),Le)),ne.registerSerializer(Te,new fi(((e,t)=>`${e.X}, ${e.Y}`),Te)),ne.registerSerializer(Me,new fi(((e,t)=>`${e.X}, ${e.Y}, ${e.Z}`),Me)),ne.registerSerializer(ce,new yi(ce)),ne.registerSerializer(Ie,new vi(((e,t)=>`${t.lookbehind??""}(${e})`),Ie)),ne.registerSerializer($e,new vi(e,$e)),ne.registerSerializer(Ne,new vi(e,Ne)),ne.registerSerializer(Ae,new vi(e,Ae))}(),customElements.define("ueb-color-handler",Rt),st.registerElement("ueb-color-handler",Rt),customElements.define("ueb-input",jt),st.registerElement("ueb-input",jt),customElements.define("ueb-link",mt),st.registerElement("ueb-link",mt),customElements.define("ueb-node",Ht),st.registerElement("ueb-node",Ht),customElements.define("ueb-pin",hi),st.registerElement("ueb-pin",hi),customElements.define("ueb-selector",gi),st.registerElement("ueb-selector",gi),customElements.define("ueb-ui-slider",Ft),st.registerElement("ueb-ui-slider",Ft),customElements.define("ueb-window",bi),st.registerElement("ueb-window",bi);export{Ot as Blueprint,te as Configuration,mt as LinkElement,Ht as NodeElement,ae as Utility}; + */const ti=(e=>(...t)=>({_$litDirective$:e,values:t}))(class extends ei{constructor(e){var t;if(super(e),e.type!==Qt||"style"!==e.name||(null===(t=e.strings)||void 0===t?void 0:t.length)>2)throw Error("The `styleMap` directive must be used in the `style` attribute and must be the only part in the attribute.")}render(e){return Object.keys(e).reduce(((t,i)=>{const n=e[i];return null==n?t:t+`${i=i.replace(/(?:^(webkit|moz|ms|o)|)(?=[A-Z])/g,"-$&").toLowerCase()}:${n};`}),"")}update(e,[t]){const{style:i}=e.element;if(void 0===this.vt){this.vt=new Set;for(const e in t)this.vt.add(e);return this.render(t)}this.vt.forEach((e=>{null==t[e]&&(this.vt.delete(e),e.includes("-")?i.removeProperty(e):i[e]="")}));for(const e in t){const n=t[e];null!=n&&(this.vt.add(e),e.includes("-")?i.setProperty(e,n):i[e]=n)}return H}});class ii extends yt{toggleAdvancedDisplayHandler;getDraggableElement(){return this.element.querySelector(".ueb-window-top")}createDraggableObject(){return new bt(this.element,this.blueprint,{draggableElement:this.getDraggableElement(),ignoreScale:!0,ignoreTranslateCompensate:!1,movementSpace:this.blueprint,stepSize:1})}setup(){const e=this.blueprint.template.gridLeftVisibilityBoundary(),t=this.blueprint.template.gridTopVisibilityBoundary();this.element.locationX=this.blueprint.scaleCorrectReverse(this.blueprint.mousePosition[0]-e),this.element.locationY=this.blueprint.scaleCorrectReverse(this.blueprint.mousePosition[1]-t),this.element.updateComplete.then((()=>{const e=this.blueprint.getBoundingClientRect();this.element.locationX+this.element.sizeX>e.width&&(this.element.locationX=e.width-this.element.sizeX),this.element.locationX=Math.max(0,this.element.locationX),this.element.locationY+this.element.sizeY>e.height&&(this.element.locationY=e.height-this.element.sizeY),this.element.locationY=Math.max(0,this.element.locationY)}))}render(){return $`
${this.renderWindowName()}
${$e.close}
${this.renderContent()}
`}renderWindowName(){return $`Window`}renderContent(){return $``}apply(){this.element.dispatchEvent(new CustomEvent(te.windowApplyEventName)),this.element.remove()}cancel(){this.element.dispatchEvent(new CustomEvent(te.windowCancelEventName)),this.element.remove()}}class ni extends ii{#Ve;#ze;#Ie;#Re;#_e;#Fe;#Ge;#je;#Ue;#Ke;#We=e=>{const t=ae.clearHTMLWhitespace(e.target.innerHTML),i=parseInt(t,16);isNaN(i)||(this.color.setFromRGBANumber(i),this.element.requestUpdate())};#Ze=e=>{const t=ae.clearHTMLWhitespace(e.target.innerHTML),i=parseInt(t,16);isNaN(i)||(this.color.setFromSRGBANumber(i),this.element.requestUpdate())};#Xe=e=>t=>{"Enter"==t.code&&(t.preventDefault(),e(t))};#ae=new Ce;get color(){return this.#ae}set color(e){e.toNumber()!=this.color?.toNumber()&&(this.element.requestUpdate("color",this.#ae),this.#ae=e)}#Ye=new Ce;get fullColor(){return this.#Ye}#qe;get initialColor(){return this.#qe}#Je=new Ce;#Qe(e,t,i=!1){const n=this.color.toRGBAString(),s=`${n.substring(0,2*e)}${t}${n.substring(2+2*e)}`;return i?`${s.substring(0,6)}FF`:s}initialize(e){super.initialize(e),this.#qe=this.element.windowOptions.getPinColor(),this.color.setFromHSVA(this.initialColor.H.value,this.initialColor.S.value,this.initialColor.V.value,this.initialColor.A.value),this.fullColor.setFromHSVA(this.color.H.value,1,1,1)}firstUpdated(e){this.#Ve=this.element.querySelector(".ueb-color-picker-wheel ueb-color-handler"),this.#ze=this.element.querySelector(".ueb-color-picker-saturation ueb-ui-slider"),this.#Ie=this.element.querySelector(".ueb-color-picker-value ueb-ui-slider"),this.#Re=this.element.querySelector(".ueb-color-picker-r ueb-ui-slider"),this.#_e=this.element.querySelector(".ueb-color-picker-g ueb-ui-slider"),this.#Fe=this.element.querySelector(".ueb-color-picker-b ueb-ui-slider"),this.#Ge=this.element.querySelector(".ueb-color-picker-a ueb-ui-slider"),this.#je=this.element.querySelector(".ueb-color-picker-h ueb-ui-slider"),this.#Ue=this.element.querySelector(".ueb-color-picker-s ueb-ui-slider"),this.#Ke=this.element.querySelector(".ueb-color-picker-v ueb-ui-slider"),this.#Ve.template.locationChangeCallback=(e,t)=>{this.color.setFromWheelLocation(e,t,this.color.V.value,this.color.A.value),this.fullColor.setFromHSVA(this.color.H.value,1,1,1),this.element.requestUpdate()},this.#ze.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,t,this.color.V.value,this.color.A.value),this.element.requestUpdate()},this.#Ie.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,this.color.S.value,t,this.color.A.value),this.element.requestUpdate()},this.#Re.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(e,this.color.G.value,this.color.B.value,this.color.A.value),this.element.requestUpdate()},this.#_e.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(this.color.R.value,e,this.color.B.value,this.color.A.value),this.element.requestUpdate()},this.#Fe.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(this.color.R.value,this.color.G.value,e,this.color.A.value),this.element.requestUpdate()},this.#Ge.template.locationChangeCallback=(e,t)=>{this.color.setFromRGBA(this.color.R.value,this.color.G.value,this.color.B.value,e),this.element.requestUpdate()},this.#je.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(e,this.color.S.value,this.color.V.value,this.color.A.value),this.element.requestUpdate()},this.#Ue.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,e,this.color.V.value,this.color.A.value),this.element.requestUpdate()},this.#Ke.template.locationChangeCallback=(e,t)=>{this.color.setFromHSVA(this.color.H.value,this.color.S.value,e,this.color.A.value),this.element.requestUpdate()}}renderSlider(e){let t="",i=0,n="";const s=e=>`linear-gradient(to right, #${this.#Qe(e,"00",!0)}, #${this.#Qe(e,"ff",!0)})`;switch(e){case 0:t="r",i=this.color.R.value,n=s(e);break;case 1:t="g",i=this.color.G.value,n=s(e);break;case 2:t="b",i=this.color.B.value,n=s(e);break;case 3:t="a",i=this.color.A.value,n=`${te.alphaPattern}, ${s(e)}`;break;case 4:t="h",i=360*this.color.H.value,n="linear-gradient(to right, #f00 0%, #ff0 16.666%, #0f0 33.333%, #0ff 50%, #00f 66.666%, #f0f 83.333%, #f00 100%)";break;case 5:t="s",i=this.color.S.value,n=`linear-gradient(to right,#${this.#Je.setFromHSVA(this.color.H.value,0,this.color.V.value,1),this.#Je.toRGBAString()},#${this.#Je.setFromHSVA(this.color.H.value,1,this.color.V.value,1),this.#Je.toRGBAString()})`;break;case 6:t="v",i=this.color.V.value,n=`linear-gradient(to right, #000, #${this.fullColor.toRGBAString()})`}return n=`background: ${n};`,$`
${t.toUpperCase()}
`}renderContent(){const e=2*this.color.H.value*Math.PI,t={"--ueb-color-r":this.color.R.toString(),"--ueb-color-g":this.color.G.toString(),"--ueb-color-b":this.color.B.toString(),"--ueb-color-a":this.color.A.toString(),"--ueb-color-h":this.color.H.toString(),"--ueb-color-s":this.color.S.toString(),"--ueb-color-v":this.color.V.toString(),"--ueb-color-wheel-x":100*(this.color.S.value*Math.cos(e)*.5+.5)+"%","--ueb-color-wheel-y":100*(this.color.S.value*Math.sin(e)*.5+.5)+"%"},i=this.color.toRGBAString(),n=this.color.toSRGBAString(),s=this.fullColor.toRGBAString();return $`
Old
New
Advanced
${this.renderSlider(0)} ${this.renderSlider(1)} ${this.renderSlider(2)} ${this.renderSlider(3)}
${this.renderSlider(4)} ${this.renderSlider(5)} ${this.renderSlider(6)}
Hex Linear
Hex sRGB
${te.windowApplyButtonText}
${te.windowCancelButtonText}
`}renderWindowName(){return $`${te.colorWindowName}`}}class si extends Tt{#et;#tt=e=>{e.preventDefault(),this.blueprint.setFocused(!0),this.#et=rt.getConstructor("ueb-window").newObject({type:new ni,windowOptions:{getPinColor:()=>this.element.defaultValue,setPinColor:e=>this.element.setDefaultValue(e)}}),this.blueprint.append(this.#et);const t=()=>{this.element.setDefaultValue(this.#et.template.color)},i=()=>{this.#et.removeEventListener(te.windowApplyEventName,t),this.#et.removeEventListener(te.windowCloseEventName,i),this.#et=null};this.#et.addEventListener(te.windowApplyEventName,t),this.#et.addEventListener(te.windowCloseEventName,i)};renderInput(){return $``}}class ri extends Xt{static singleLineInput=!0}class ai extends Yt{setDefaultValue(e=[],t=e){this.element.setDefaultValue(e[0])}renderInput(){return $`
`}}class oi extends Tt{renderIcon(){return $e.referencePin}}class li extends Yt{#it(){return ae.minDecimals(this.element.getDefaultValue()?.R??0)}#nt(){return ae.minDecimals(this.element.getDefaultValue()?.P??0)}#st(){return ae.minDecimals(this.element.getDefaultValue()?.Y??0)}setDefaultValue(e=[],t=e){const i=this.element.getDefaultValue(!0);if(!(i instanceof Le))throw new TypeError("Expected DefaultValue to be a RotatorEntity");i.R=e[0],i.P=e[1],i.Y=e[2],this.element.requestUpdate("DefaultValue",i)}renderInput(){return $`
X
Y
Z
`}}class ui extends Xt{}class ci extends Yt{#rt(){return ae.minDecimals(this.element.getDefaultValue()?.X??0)}#st(){return ae.minDecimals(this.element.getDefaultValue()?.Y??0)}setDefaultValue(e,t){const i=this.element.getDefaultValue(!0);if(!(i instanceof Te))throw new TypeError("Expected DefaultValue to be a Vector2DEntity");i.X=e[0],i.Y=e[1],this.element.requestUpdate("DefaultValue",i)}renderInput(){return $`
X
Y
`}}class hi extends Yt{#rt(){return ae.minDecimals(this.element.getDefaultValue()?.X??0)}#st(){return ae.minDecimals(this.element.getDefaultValue()?.Y??0)}#at(){return ae.minDecimals(this.element.getDefaultValue()?.Z??0)}setDefaultValue(e,t){const i=this.element.getDefaultValue(!0);if(!(i instanceof Me))throw new TypeError("Expected DefaultValue to be a VectorEntity");i.X=e[0],i.Y=e[1],i.Z=e[2],this.element.requestUpdate("DefaultValue",i)}renderInput(){return $`
X
Y
Z
`}}class di extends et{static#ot={"/Script/CoreUObject.LinearColor":si,"/Script/CoreUObject.Rotator":li,"/Script/CoreUObject.Vector":hi,"/Script/CoreUObject.Vector2D":ci,bool:Wt,byte:qt,int:qt,int64:Jt,MUTABLE_REFERENCE:oi,name:ri,real:ai,string:ui};static properties={pinId:{type:ge,converter:{fromAttribute:(e,t)=>e?Ge.grammar.Guid.parse(e).value:null,toAttribute:(e,t)=>e?.toString()},attribute:"data-id",reflect:!0},pinType:{type:String,attribute:"data-type",reflect:!0},advancedView:{type:String,attribute:"data-advanced-view",reflect:!0},color:{type:Ce,converter:{fromAttribute:(e,t)=>e?Ge.grammar.LinearColorFromAnyColor.parse(e).value:null,toAttribute:(e,t)=>e?ae.printLinearColor(e):null},attribute:"data-color",reflect:!0},defaultValue:{type:String,attribute:!1},isLinked:{type:Boolean,converter:ae.booleanConverter,attribute:"data-linked",reflect:!0},pinDirection:{type:String,attribute:"data-direction",reflect:!0}};nodeElement;static getTypeTemplate(e){if(e.PinType.bIsReference&&!e.PinType.bIsConst)return di.#ot.MUTABLE_REFERENCE;if("exec"===e.getType())return Zt;let t;return e.isInput()&&(t=di.#ot[e.getType()]),t??Tt}static newObject(e=new De,t=new(di.getTypeTemplate(e)),i){const n=new di;return n.initialize(e,t,i),n}initialize(e=new De,t=new(di.getTypeTemplate(e)),i){super.initialize(e,t),this.pinId=this.entity.PinId,this.pinType=this.entity.getType(),this.advancedView=this.entity.bAdvancedView,this.defaultValue=this.entity.getDefaultValue(),this.color=di.properties.color.converter.fromAttribute(this.getColor().toString()),this.isLinked=!1,this.pinDirection=e.isInput()?"input":e.isOutput()?"output":"hidden",this.nodeElement=i}setup(){super.setup(),this.nodeElement=this.closest("ueb-node")}createPinReference(){return new Pe({objectName:this.nodeElement.getNodeName(),pinGuid:this.getPinId()})}getPinId(){return this.entity.PinId}getPinName(){return this.entity.PinName}getPinDisplayName(){return this.entity.getDisplayName()}getColor(){return this.entity.pinColor()}isInput(){return this.entity.isInput()}isOutput(){return this.entity.isOutput()}getLinkLocation(){return this.template.getLinkLocation()}getNodeElement(){return this.nodeElement}getLinks(){return this.entity.LinkedTo??[]}getDefaultValue(e=!1){return this.defaultValue=this.entity.getDefaultValue(e)}setDefaultValue(e){this.entity.DefaultValue=e,this.defaultValue=e}sanitizeLinks(e=[]){this.entity.LinkedTo=this.entity.LinkedTo?.filter((t=>{let i=this.blueprint.getPin(t);if(i){if(e.length&&!e.includes(i.nodeElement))return!1;let t=this.blueprint.getLink(this,i);t||(t=rt.getConstructor("ueb-link").newObject(this,i),this.blueprint.addGraphElement(t))}return i})),this.isLinked=this.entity.isLinked()}linkTo(e){const t=this.createPinReference();!this.isLinked||!this.isOutput()||"exec"!==this.pinType&&"exec"!==e.pinType||this.getLinks().some((e=>t.equals(e)))||this.unlinkFromAll(),this.entity.linkTo(e.getNodeElement().getNodeName(),e.entity)&&(this.isLinked=this.entity.isLinked(),this.nodeElement?.template.linksChanged())}unlinkFrom(e,t=!0){this.entity.unlinkFrom(e.getNodeElement().getNodeName(),e.entity)&&(this.isLinked=this.entity.isLinked(),this.nodeElement?.template.linksChanged(),t&&this.blueprint.getLink(this,e)?.remove())}unlinkFromAll(){const e=this.getLinks().length;this.getLinks().map((e=>this.blueprint.getPin(e))).forEach((e=>this.unlinkFrom(e))),e&&this.nodeElement?.template.linksChanged()}redirectLink(e,t){const i=this.getLinks().findIndex((t=>t.objectName.toString()==e.getNodeElement().getNodeName()&&t.pinGuid.valueOf()==e.entity.PinId.valueOf()));return i>=0&&(this.entity.LinkedTo[i]=t,!0)}}class pi{constructor(e=(e=>e),t=null){this.array=new Uint32Array(t),this.comparisonValueSupplier=e,this.length=0,this.currentPosition=0}get(e){return e>=0&&e=0&&this.currentPosition=0&&this.currentPosition0?this.get(this.currentPosition-1):null}getPrevValue(){return this.currentPosition>0?this.comparisonValueSupplier(this.get(this.currentPosition-1)):Number.MIN_SAFE_INTEGER}shiftLeft(e,t=1){this.array.set(this.array.subarray(e+t),e)}shiftRight(e,t=1){this.array.set(this.array.subarray(e,-t),e+t)}}class mi{constructor(e,t,i,n){this.initialPosition=e,this.finalPosition=e,this.metadata=new Array(t.length),this.primaryOrder=new pi((e=>this.metadata[e].primaryBoundary)),this.secondaryOrder=new pi((e=>this.metadata[e].secondaryBoundary)),this.selectFunc=n,this.rectangles=t,this.primaryOrder.reserve(this.rectangles.length),this.secondaryOrder.reserve(this.rectangles.length),t.forEach(((e,t)=>{let s={primaryBoundary:this.initialPosition[0],secondaryBoundary:this.initialPosition[1],rectangle:t,onSecondaryAxis:!1};this.metadata[t]=s,n(e,!1);const r=i(e);this.initialPosition[1]{if(this.metadata[i].onSecondaryAxis)this.selectFunc(this.rectangles[i],n);else if(n){this.secondaryOrder.insert(i,e[1]);const n=this.metadata[i].secondaryBoundary;Math.sign(e[1]-n)==t[1]&&Math.sign(n-this.initialPosition[1])==t[1]&&this.selectFunc(this.rectangles[i],!0)}else this.selectFunc(this.rectangles[i],!1),this.secondaryOrder.remove(i);this.computeBoundaries(),this.selectTo(e)};e[0]this.boundaries.primaryN.v&&e[0]this.boundaries.primaryP.v&&(++this.primaryOrder.currentPosition,i(this.boundaries.primaryP.i,this.initialPosition[0]{this.selectFunc(this.rectangles[t],i),this.computeBoundaries(),this.selectTo(e)};e[1]this.boundaries.secondaryN.v&&e[1]this.boundaries.secondaryP.v&&(++this.secondaryOrder.currentPosition,n(this.boundaries.secondaryP.i,this.initialPosition[1]vi.#lt[e],toAttribute:(e,t)=>Object.entries(vi.#lt).find((([t,i])=>e.constructor===i))?.[0]}}};static newObject(e={},t=e.type??new ii){const i=new vi;return i.initialize(e,t),i}initialize(e={},t=e.type??new ii){e.windowOptions??={},this.type=e.type,this.windowOptions=e.windowOptions,super.initialize(e,t)}computeSizes(){const e=this.getBoundingClientRect();this.sizeX=e.width,this.sizeY=e.height}cleanup(){super.cleanup(),this.acknowledgeClose()}acknowledgeClose(){let e=new CustomEvent(te.windowCloseEventName);this.dispatchEvent(e)}}class fi extends Ge{constructor(e,t,i,n,s,r,a){e=e??(e=>`(${e})`),super(t,i,n,s,r,a),this.wrap=e}read(e){const t=Fe.getGrammarForType(Ge.grammar,this.entityType).parse(e);if(!t.status)throw new Error(`Error when trying to parse the entity ${this.entityType.prototype.constructor.name}.`);return t.value}write(e,t,i=!1){return this.wrap(this.subWrite(e,[],t,i),t)}}class yi extends fi{#ut;constructor(e,t){super(void 0,t),this.#ut=e}write(e,t,i=!1){return this.#ut(t,i)}}class wi extends fi{constructor(e){super(void 0,e)}write(e,t,i){return i||t.constructor!==String?ae.escapeString(t.toString()):`"${ae.escapeString(t.toString())}"`}}!function(){const e=e=>`(${e})`;ne.registerSerializer(null,new yi(((e,t)=>"()"),null)),ne.registerSerializer(Array,new yi(((e,t)=>`(${e.map((e=>ne.getSerializer(ae.getType(e)).serialize(e,t)+",")).join("")})`),Array)),ne.registerSerializer(BigInt,new wi(BigInt)),ne.registerSerializer(Boolean,new yi(((e,t)=>e?t?"true":"True":t?"false":"False"),Boolean)),ne.registerSerializer(ue,new wi(ue)),ne.registerSerializer(he,new wi(he)),ne.registerSerializer(ve,new fi(e,ve)),ne.registerSerializer(ge,new wi(ge)),ne.registerSerializer(fe,new wi(fe)),ne.registerSerializer(ye,new wi(ye)),ne.registerSerializer(le,new wi(le)),ne.registerSerializer(de,new fi((e=>`${de.lookbehind}(${e})`),de,"",", ",!1,"",(e=>""))),ne.registerSerializer(we,new fi(e,we)),ne.registerSerializer(Ce,new fi(e,Ce)),ne.registerSerializer(pe,new fi((e=>`${pe.lookbehind}(${e})`),pe,"",", ",!1,"",(e=>""))),ne.registerSerializer(Se,new fi(e,Se)),ne.registerSerializer(Number,new wi(Number)),ne.registerSerializer(Oe,new je),ne.registerSerializer(be,new yi((e=>(e.type??"")+(e.path?e.type?`'"${e.path}"'`:`"${e.path}"`:"")),be)),ne.registerSerializer(xe,new wi(xe)),ne.registerSerializer(De,new fi((e=>`${De.lookbehind} (${e})`),De,"",",",!0)),ne.registerSerializer(Pe,new fi((e=>e),Pe,""," ",!1,"",(e=>""))),ne.registerSerializer(ke,new fi(e,ke)),ne.registerSerializer(Ee,new wi(Ee)),ne.registerSerializer(Le,new fi(e,Le)),ne.registerSerializer(String,new yi(((e,t)=>t?ae.escapeString(e):`"${ae.escapeString(e)}"`),String)),ne.registerSerializer(Ne,new yi(((e,t)=>`${e.P}, ${e.Y}, ${e.R}`),Ne)),ne.registerSerializer(Ae,new yi(((e,t)=>`${e.X}, ${e.Y}`),Ae)),ne.registerSerializer(Be,new yi(((e,t)=>`${e.X}, ${e.Y}, ${e.Z}`),Be)),ne.registerSerializer(ce,new wi(ce)),ne.registerSerializer(Re,new fi(((e,t)=>`${t.lookbehind??""}(${e})`),Re)),ne.registerSerializer(He,new fi(e,He)),ne.registerSerializer(Te,new fi(e,Te)),ne.registerSerializer(Me,new fi(e,Me))}(),customElements.define("ueb-color-handler",_t),rt.registerElement("ueb-color-handler",_t),customElements.define("ueb-input",Ut),rt.registerElement("ueb-input",Ut),customElements.define("ueb-link",gt),rt.registerElement("ueb-link",gt),customElements.define("ueb-node",Ot),rt.registerElement("ueb-node",Ot),customElements.define("ueb-pin",di),rt.registerElement("ueb-pin",di),customElements.define("ueb-selector",bi),rt.registerElement("ueb-selector",bi),customElements.define("ueb-ui-slider",Gt),rt.registerElement("ueb-ui-slider",Gt),customElements.define("ueb-window",vi),rt.registerElement("ueb-window",vi);export{Vt as Blueprint,te as Configuration,gt as LinkElement,Ot as NodeElement,ae as Utility}; diff --git a/js/Configuration.js b/js/Configuration.js index 158300f..7746e17 100755 --- a/js/Configuration.js +++ b/js/Configuration.js @@ -76,6 +76,7 @@ export default class Configuration { componentBoundEvent: "/Script/BlueprintGraph.K2Node_ComponentBoundEvent", customEvent: "/Script/BlueprintGraph.K2Node_CustomEvent", doN: "/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:Do N", + doOnce: "/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:DoOnce", dynamicCast: "/Script/BlueprintGraph.K2Node_DynamicCast", enum: "/Script/CoreUObject.Enum", enumLiteral: "/Script/BlueprintGraph.K2Node_EnumLiteral", @@ -98,6 +99,7 @@ export default class Configuration { makeArray: "/Script/BlueprintGraph.K2Node_MakeArray", makeMap: "/Script/BlueprintGraph.K2Node_MakeMap", makeSet: "/Script/BlueprintGraph.K2Node_MakeSet", + multiGate: "/Script/BlueprintGraph.K2Node_MultiGate", pawn: "/Script/Engine.Pawn", promotableOperator: "/Script/BlueprintGraph.K2Node_PromotableOperator", reverseForEachLoop: "/Engine/EditorBlueprintResources/StandardMacros.StandardMacros:ReverseForEachLoop", diff --git a/js/SVGIcon.js b/js/SVGIcon.js index 570b454..47fd38f 100644 --- a/js/SVGIcon.js +++ b/js/SVGIcon.js @@ -64,12 +64,20 @@ export default class SVGIcon { ` static doN = html` - + ` + static doOnce = html` + + + + + + ` + static enum = html` diff --git a/js/entity/FormatTextEntity.js b/js/entity/FormatTextEntity.js index e442f8a..3349d2a 100644 --- a/js/entity/FormatTextEntity.js +++ b/js/entity/FormatTextEntity.js @@ -7,7 +7,9 @@ export default class FormatTextEntity extends IEntity { static lookbehind = "LOCGEN_FORMAT_NAMED" static attributes = { - value: [new UnionType(LocalizedTextEntity, InvariantTextEntity, FormatTextEntity)], + value: { + type: [new UnionType(LocalizedTextEntity, InvariantTextEntity, FormatTextEntity)] + }, } static { @@ -18,4 +20,4 @@ export default class FormatTextEntity extends IEntity { super(values) /** @type {String} */ this.value } -} 1 +} diff --git a/js/entity/ObjectEntity.js b/js/entity/ObjectEntity.js index 5c6a9e0..c4dae3d 100755 --- a/js/entity/ObjectEntity.js +++ b/js/entity/ObjectEntity.js @@ -540,6 +540,7 @@ export default class ObjectEntity extends IEntity { case Configuration.nodeType.executionSequence: case Configuration.nodeType.ifThenElse: case Configuration.nodeType.macro: + case Configuration.nodeType.multiGate: return Configuration.nodeColors.gray case Configuration.nodeType.dynamicCast: return Configuration.nodeColors.turquoise @@ -557,22 +558,27 @@ export default class ObjectEntity extends IEntity { switch (this.getType()) { case Configuration.nodeType.customEvent: return SVGIcon.event case Configuration.nodeType.doN: return SVGIcon.doN + case Configuration.nodeType.doOnce: return SVGIcon.doOnce case Configuration.nodeType.dynamicCast: return SVGIcon.cast case Configuration.nodeType.enumLiteral: return SVGIcon.enum case Configuration.nodeType.event: return SVGIcon.event - case Configuration.nodeType.executionSequence: return SVGIcon.sequence - case Configuration.nodeType.forEachElementInEnum: return SVGIcon.loop - case Configuration.nodeType.forEachLoop: return SVGIcon.forEachLoop - case Configuration.nodeType.forEachLoopWithBreak: return SVGIcon.forEachLoop - case Configuration.nodeType.forLoop: return SVGIcon.loop - case Configuration.nodeType.forLoopWithBreak: return SVGIcon.loop + case Configuration.nodeType.executionSequence: + case Configuration.nodeType.multiGate: + return SVGIcon.sequence + case Configuration.nodeType.forEachElementInEnum: + case Configuration.nodeType.forLoop: + case Configuration.nodeType.forLoopWithBreak: + case Configuration.nodeType.whileLoop: + return SVGIcon.loop + case Configuration.nodeType.forEachLoop: + case Configuration.nodeType.forEachLoopWithBreak: + return SVGIcon.forEachLoop case Configuration.nodeType.ifThenElse: return SVGIcon.branchNode case Configuration.nodeType.isValid: return SVGIcon.questionMark case Configuration.nodeType.makeArray: return SVGIcon.makeArray case Configuration.nodeType.makeMap: return SVGIcon.makeMap case Configuration.nodeType.makeSet: return SVGIcon.makeSet case Configuration.nodeType.select: return SVGIcon.select - case Configuration.nodeType.whileLoop: return SVGIcon.loop } if (this.nodeDisplayName().startsWith("Break")) { return SVGIcon.breakStruct diff --git a/js/entity/PinEntity.js b/js/entity/PinEntity.js index 73b78cb..eb448ed 100755 --- a/js/entity/PinEntity.js +++ b/js/entity/PinEntity.js @@ -1,6 +1,7 @@ import ByteEntity from "./ByteEntity" import Configuration from "../Configuration" import EnumEntity from "./EnumEntity" +import FormatTextEntity from "./FormatTextEntity" import FunctionReferenceEntity from "./FunctionReferenceEntity" import GuidEntity from "./GuidEntity" import IEntity from "./IEntity" @@ -57,7 +58,7 @@ export default class PinEntity extends IEntity { }, PinName: "", PinFriendlyName: { - type: new UnionType(LocalizedTextEntity, String), + type: new UnionType(LocalizedTextEntity, FormatTextEntity, String), showDefault: false, }, PinToolTip: { diff --git a/js/serialization/Grammar.js b/js/serialization/Grammar.js index ee9dbc1..af32c31 100755 --- a/js/serialization/Grammar.js +++ b/js/serialization/Grammar.js @@ -1,6 +1,7 @@ // @ts-nocheck import ByteEntity from "../entity/ByteEntity" import EnumEntity from "../entity/EnumEntity" +import FormatTextEntity from "../entity/FormatTextEntity" import FunctionReferenceEntity from "../entity/FunctionReferenceEntity" import GuidEntity from "../entity/GuidEntity" import IdentifierEntity from "../entity/IdentifierEntity" @@ -89,33 +90,6 @@ export default class Grammar { /** @param {Grammar} r */ static getGrammarForType(r, attribute, defaultGrammar = r.AttributeAnyValue) { - if (attribute.constructor === Object) { - attribute = /** @type {AttributeInformation} */(attribute) - let type = attribute.type - let result - if (type instanceof Array) { - result = Grammar.getGrammarForType(r, type[0]) - .trim(P.optWhitespace) - .sepBy(P.string(",")) - .skip(P.regex(/,?\s*/)) - .wrap(P.string("("), P.string(")")) - } else if (type instanceof UnionType) { - result = type.types - .map(v => Grammar.getGrammarForType(r, Utility.getType(v))) - .reduce((accum, cur) => !cur || accum === r.AttributeAnyValue - ? r.AttributeAnyValue - : accum.or(cur)) - } else { - result = Grammar.getGrammarForType(r, type, defaultGrammar) - } - if (attribute.serialized && !(type instanceof String)) { - result = result.wrap(P.string('"'), P.string('"')) - } - if (attribute.nullable) { - result = result.or(r.Null) - } - return result - } switch (attribute) { case BigInt: return r.BigInt @@ -125,6 +99,8 @@ export default class Grammar { return r.Byte case EnumEntity: return r.Enum + case FormatTextEntity: + return r.FormatText case FunctionReferenceEntity: return r.FunctionReference case GuidEntity: @@ -177,9 +153,35 @@ export default class Grammar { return r.Vector2D case VectorEntity: return r.Vector - default: - return defaultGrammar } + let result = defaultGrammar + const type = attribute.constructor === Object + ? attribute.type + : attribute.constructor + if (type instanceof Array) { + result = Grammar.getGrammarForType(r, type[0]) + .trim(P.optWhitespace) + .sepBy(P.string(",")) + .skip(P.regex(/,?\s*/)) + .wrap(P.string("("), P.string(")")) + } else if (type instanceof UnionType) { + result = type.types + .map(v => Grammar.getGrammarForType(r, v)) + .reduce((accum, cur) => !cur || accum === r.AttributeAnyValue + ? r.AttributeAnyValue + : accum.or(cur)) + } else if (attribute.constructor === Object) { + result = Grammar.getGrammarForType(r, type, defaultGrammar) + } + if (attribute.constructor === Object) { + if (attribute.serialized && type.constructor !== String) { + result = result.wrap(P.string('"'), P.string('"')) + } + if (attribute.nullable) { + result = result.or(r.Null) + } + } + return result } /** @param {Grammar} r */ @@ -222,7 +224,7 @@ export default class Grammar { /** * @param {Grammar} r * @param {EntityConstructor} entityType - * @param {Boolean | Number} acceptUnknownKeys can be anumber to specify the limit or true, to let it be a reasonable value + * @param {Boolean | Number} acceptUnknownKeys Number to specify the limit or true, to let it be a reasonable value */ static createEntityGrammar = (r, entityType, acceptUnknownKeys = true) => P.seqMap( @@ -366,19 +368,20 @@ export default class Grammar { namespace: matchResult[1], key: matchResult[2], value: matchResult[3] - } - ) + }) ) /** @param {Grammar} r */ InvariantText = r => Grammar.regexMap( - new RegExp( - String.raw`${InvariantTextEntity.lookbehind}\s*\("(${Grammar.Regex.InsideString.source})"\)`, - matchResult => new InvariantTextEntity({ value: matchResult[1] }) - ) + new RegExp(String.raw`${InvariantTextEntity.lookbehind}\s*\("(${Grammar.Regex.InsideString.source})"\)`), + matchResult => new InvariantTextEntity({ value: matchResult[1] }) ) + FormatText = r => P.string(FormatTextEntity.lookbehind).then(P.optWhitespace).then( + Grammar.getGrammarForType(r, FormatTextEntity.attributes.value).wrap(P.string("("), P.string(")")) + ) + /** @param {Grammar} r */ AttributeAnyValue = r => P.alt( // Remember to keep the order, otherwise parsing might fail