diff --git a/dist/ueblueprint.js b/dist/ueblueprint.js index 262dfef..ed97b87 100755 --- a/dist/ueblueprint.js +++ b/dist/ueblueprint.js @@ -246,6 +246,29 @@ class IInput { } } +/** + * @typedef {import("./IEntity").default} IEntity + */ + +class CalculatedType { + + #f + + /** + * @param {Function} f + */ + constructor(f) { + this.#f = f; + } + + /** + * @param {IEntity} entity + */ + calculate(entity) { + return this.#f(entity) + } +} + class Observable { /** @type {Map} */ @@ -352,33 +375,6 @@ class Observable { } } -class SerializedType { - - #types - get types() { - return this.#types - } - set types(v) { - this.#types = v; - } - - #stringFallback - get stringFallback() { - return this.#stringFallback - } - set stringFallback(v) { - this.#stringFallback = v; - } - - constructor([...acceptedTypes], stringFallback = true) { - this.#types = [...new Set([ - ...acceptedTypes, - ...(stringFallback ? [String] : []) - ])]; - this.#stringFallback = stringFallback; - } -} - /** * @template T */ @@ -410,13 +406,22 @@ class TypeInitialization { this.#value = v; } + /** @type {Boolean} */ + #serialized + get serialized() { + return this.#serialized + } + set serialized(v) { + this.#serialized = v; + } + static sanitize(value, targetType) { if (targetType === undefined) { targetType = value?.constructor; } if ( targetType - && targetType !== SerializedType + // value is not of type targetType && !(value?.constructor === targetType || value instanceof targetType) ) { value = new targetType(value); @@ -428,28 +433,32 @@ class TypeInitialization { } /** - * @typedef {(new () => T) | SerializedType | StringConstructor | NumberConstructor | BooleanConstructor} Constructor + * @typedef {(new () => T) | StringConstructor | NumberConstructor | BooleanConstructor} Constructor * @param {Constructor|Array} type * @param {Boolean} showDefault * @param {any} value + * @param {Boolean} serialized */ - constructor(type, showDefault = true, value = undefined) { + constructor(type, showDefault = true, value = undefined, serialized = false) { if (value === undefined) { if (type instanceof Array) { value = []; - } else if (type instanceof SerializedType) { + } else if (serialized) { value = ""; } else { value = TypeInitialization.sanitize(new type()); } } - this.#showDefault = showDefault; this.#type = type; + this.#showDefault = showDefault; + this.#value = value; + this.#serialized = serialized; } } /** * @typedef {import("./entity/LinearColorEntity").default} LinearColorEntity + * @typedef {import("./entity/IEntity").default} IEntity */ class Utility { @@ -507,6 +516,26 @@ class Utility { return location } + /** + * @param {IEntity} + * @param {Object} target Object holding the data + * @param {String[]} keys The chained keys to access from object in order to get the value + * @param {Boolean} defaultValue Value to return in case from doesn't have it + * @returns {any} The value in from corresponding to the keys or defaultValue otherwise + */ + static isSerialized(entity, keys, propertyDefinition = Utility.objectGet(entity.constructor.attributes, keys)) { + if (propertyDefinition instanceof CalculatedType) { + return Utility.isSerialized(entity, keys, propertyDefinition.calculate(entity)) + } + if (propertyDefinition instanceof TypeInitialization) { + if (propertyDefinition.serialized) { + return true + } + return Utility.isSerialized(entity, keys, propertyDefinition.type) + } + return false + } + /** * Gets a value from an object, gives defaultValue in case of failure * @param {Object} target Object holding the data @@ -575,6 +604,9 @@ class Utility { } static getType(value) { + if (value === null) { + return null + } let constructor = value?.constructor; switch (constructor) { case TypeInitialization: @@ -654,8 +686,9 @@ class Utility { */ static encodeString(value, input = false) { return value + .replaceAll('"', '\\"') // Escape " + .replaceAll("\n", "\\n") // Replace newline with \n .replaceAll("\u00A0", " ") // Replace special space symbol - .replaceAll("\n", String.raw`\n`) // Replace newline with \n } /** @@ -663,8 +696,9 @@ class Utility { */ static decodeString(value, input = false) { return value - .replaceAll(" ", "\u00A0") // Replace special space symbol - .replaceAll(String.raw`\n`, "\n") // Replace newline with \n + .replaceAll('\\"', '"') + .replaceAll("\\n", "\n") + .replaceAll(" ", "\u00A0") } /** @@ -689,33 +723,33 @@ class Utility { } } -/** - * @typedef {import("./IEntity").default} IEntity - */ - -class CalculatedType { - - #f - - /** - * @param {Function} f - */ - constructor(f) { - this.#f = f; - } - - /** - * @param {IEntity} entity - */ - calculate(entity) { - return this.f(entity) - } +/** @typedef {import("../entity/IEntity").default} IEntity */ +/** + * @template {IEntity} T + * @typedef {import("./ISerializer").default} ISerializer + */ + +class SerializerFactory { + + /** @type {Map>} */ + static #serializers = new Map() + + static registerSerializer(entity, object) { + SerializerFactory.#serializers.set(entity, object); + } + + /** + * @template {IEntity} T + * @param {T} entity + */ + static getSerializer(entity) { + return SerializerFactory.#serializers.get(entity) + } } class IEntity extends Observable { static attributes = {} - #showAsString = false constructor(values) { super(); @@ -730,19 +764,25 @@ class IEntity extends Observable { Object.getOwnPropertyNames(properties), Object.getOwnPropertyNames(values ?? {}) )) { - + let value = Utility.objectGet(values, [property]); let defaultValue = properties[property]; - const defaultType = Utility.getType(defaultValue); + let defaultType = Utility.getType(defaultValue); + if (defaultValue instanceof CalculatedType) { + defaultValue = defaultValue.calculate(this); + defaultType = Utility.getType(defaultValue); + } if (!(property in properties)) { - console.warn(`Property ${prefix}${property} is not defined in ${this.constructor.name}.attributes`); + console.warn( + `Property ${prefix}${property} in the serialized data is not defined in ${this.constructor.name}.properties` + ); } else if ( - defaultValue != null + !(property in values) + && defaultValue !== undefined && !(defaultValue instanceof TypeInitialization && !defaultValue.showDefault) - && !(property in values) ) { console.warn( - `${this.constructor.name} adds property ${prefix}${property} not defined in the serialized data` + `${this.constructor.name}.properties will add property ${prefix}${property} not defined in the serialized data` ); } @@ -753,34 +793,34 @@ class IEntity extends Observable { continue } - /* - * The value can either be: - * - Array: can contain multiple values, its property is assigned multiple times like (X=1, X="4"). - * - CalculatedType: the exact type depends on the previous attributes assigned to this entity. - * - TypeInitialization: contains the maximum amount of information about the attribute. - * - A type: the default value will be default constructed object without arguments. - * - A proper value. - */ - const value = Utility.objectGet(values, [property]); if (value !== undefined) { - target[property] = TypeInitialization.sanitize(value, defaultType); - // We have a value, need nothing more - continue - } - if (defaultValue instanceof CalculatedType) { - defaultValue = defaultValue.calculate(this); - defaultType = Utility.getType(defaultValue); + // Remember value can still be null + if ( + value?.constructor === String + && defaultValue instanceof TypeInitialization + && defaultValue.serialized + && defaultValue.type !== String + ) { + value = SerializerFactory.getSerializer(defaultValue.type).deserialize(value); + } + target[property] = TypeInitialization.sanitize(value, Utility.getType(defaultValue)); + continue // We have a value, need nothing more } + if (defaultValue instanceof TypeInitialization) { if (!defaultValue.showDefault) { target[property] = undefined; // Declare undefined to preserve the order of attributes continue } - defaultValue = defaultValue.value; + if (defaultValue.serialized) { + defaultValue = ""; + } else { + defaultType = defaultValue.type; + defaultValue = defaultValue.value; + } } if (defaultValue instanceof Array) { - target[property] = []; - continue + defaultValue = []; } target[property] = TypeInitialization.sanitize(defaultValue, defaultType); } @@ -794,17 +834,6 @@ class IEntity extends Observable { } defineAllAttributes(this, attributes, values); } - - isShownAsString() { - return this.#showAsString - } - - /** - * @param {Boolean} v - */ - setShowAsString(v) { - this.#showAsString = v; - } } class ObjectReferenceEntity extends IEntity { @@ -922,7 +951,7 @@ class KeyBindingEntity extends IEntity { } } -class LinearColorEntity$1 extends IEntity { +class LinearColorEntity extends IEntity { static attributes = { R: Number, @@ -969,8 +998,29 @@ class PinReferenceEntity extends IEntity { } } +class VectorEntity extends IEntity { + + static attributes = { + X: Number, + Y: Number, + Z: Number, + } +} + class PinEntity extends IEntity { + static #typeEntityMap = { + "/Script/CoreUObject.LinearColor": LinearColorEntity, + "/Script/CoreUObject.Vector": VectorEntity, + "bool": Boolean, + "exec": String, + "name": String, + "real": Number, + "string": String, + } + static get typeEntityMap() { + return PinEntity.#typeEntityMap + } static lookbehind = "Pin" static attributes = { PinId: GuidEntity, @@ -992,12 +1042,16 @@ class PinEntity extends IEntity { bSerializeAsSinglePrecisionFloat: false, }, LinkedTo: new TypeInitialization([PinReferenceEntity], false), - DefaultValue: new TypeInitialization( - new SerializedType([ - LinearColorEntity$1, - ]), - false - ), + DefaultValue: + new CalculatedType( + /** @param {PinEntity} pinEntity */ + pinEntity => new TypeInitialization( + PinEntity.typeEntityMap[pinEntity.getType()] ?? String, + false, + undefined, + true + ) + ), AutogeneratedDefaultValue: new TypeInitialization(String, false), DefaultObject: new TypeInitialization(ObjectReferenceEntity, false, null), PersistentGuid: GuidEntity, @@ -1009,6 +1063,13 @@ class PinEntity extends IEntity { bOrphanedPin: false, } + getType() { + if (this.PinType.PinCategory == "struct") { + return this.PinType.PinSubCategoryObject.path + } + return this.PinType.PinCategory + } + getDefaultValue() { return this.DefaultValue ?? "" } @@ -1070,13 +1131,6 @@ class PinEntity extends IEntity { return false } - getType() { - if (this.PinType.PinCategory == "struct") { - return this.PinType.PinSubCategoryObject.path - } - return this.PinType.PinCategory - } - getSubCategory() { return this.PinType.PinSubCategoryObject.path } @@ -1160,15 +1214,6 @@ var parsimmon_umd_min = {exports: {}}; var Parsimmon = /*@__PURE__*/getDefaultExportFromCjs(parsimmon_umd_min.exports); -class LinearColorEntity extends IEntity { - - static attributes = { - X: Number, - Y: Number, - Z: Number, - } -} - /** * @typedef {import("../entity/IEntity").default} IEntity */ @@ -1181,32 +1226,9 @@ class Grammar { static getGrammarForType(r, attributeType, defaultGrammar) { if (attributeType instanceof TypeInitialization) { - // Unpack TypeInitialization - attributeType = attributeType.type; - return Grammar.getGrammarForType(r, attributeType, defaultGrammar) - } - if (attributeType instanceof SerializedType) { - const nonStringTypes = attributeType.types.filter(t => t !== String); - let result = P.alt( - ...nonStringTypes.map(t => - Grammar.getGrammarForType(r, t).wrap(P.string('"'), P.string('"')).map( - /** - * @param {IEntity} entity - */ - entity => { - entity.setShowAsString(true); // Showing as string because it is inside a SerializedType - return entity - } - ) - ) - ); - if (nonStringTypes.length < attributeType.types.length) { - result = result.or(r.String/*.map(v => { - if (attributeType.stringFallback) { - console.log("Unrecognized value, fallback on String") - } - return v - })*/); // Separated because it cannot be wrapped into " and " + let result = Grammar.getGrammarForType(r, attributeType.type, defaultGrammar); + if (attributeType.serialized && !(attributeType.type instanceof String)) { + result = result.wrap(P.string('"'), P.string('"')); } return result } @@ -1231,9 +1253,9 @@ class Grammar { return r.InvariantText case PinReferenceEntity: return r.PinReference - case LinearColorEntity: + case VectorEntity: return r.Vector - case LinearColorEntity$1: + case LinearColorEntity: return r.LinearColor case FunctionReferenceEntity: return r.FunctionReference @@ -1300,7 +1322,12 @@ class Grammar { Null = r => P.seq(P.string("("), r.InlineOptWhitespace, P.string(")")).map(_ => null).desc("null: ()") - Boolean = r => P.alt(P.string("True"), P.string("False")).map(v => v === "True" ? true : false) + Boolean = r => P.alt( + P.string("True"), + P.string("true"), + P.string("False"), + P.string("false"), + ).map(v => v.toLocaleLowerCase() === "true" ? true : false) .desc("either True or False") HexDigit = r => P.regex(/[0-9a-fA-f]/).desc("hexadecimal digit") @@ -1403,13 +1430,13 @@ class Grammar { r.Guid, // Goes into pinGuid (objectName, _, pinGuid) => new PinReferenceEntity({ objectName: objectName, - pinGuid: pinGuid + pinGuid: pinGuid, }) ) - Vector = r => Grammar.createEntityGrammar(r, LinearColorEntity) + Vector = r => Grammar.createEntityGrammar(r, VectorEntity) - LinearColor = r => Grammar.createEntityGrammar(r, LinearColorEntity$1) + LinearColor = r => Grammar.createEntityGrammar(r, LinearColorEntity) FunctionReference = r => Grammar.createEntityGrammar(r, FunctionReferenceEntity) @@ -1459,7 +1486,7 @@ class Grammar { .string("#") .then(r.HexDigit.times(2).tie().times(3, 4)) .trim(P.optWhitespace) - .map(([R, G, B, A]) => new LinearColorEntity$1({ + .map(([R, G, B, A]) => new LinearColorEntity({ R: parseInt(R, 16) / 255, G: parseInt(G, 16) / 255, B: parseInt(B, 16) / 255, @@ -1472,12 +1499,12 @@ class Grammar { r.ColorNumber, P.string(",").skip(P.optWhitespace), r.ColorNumber.map(Number), - (R, _, G, __, B) => new LinearColorEntity$1({ + (R, _, G, __, B) => new LinearColorEntity({ R: R / 255, G: G / 255, B: B / 255, A: 1, - }), + }) ) LinearColorFromRGB = r => P.string("rgb").then( @@ -1496,7 +1523,7 @@ class Grammar { r.ColorNumber.map(Number), P.string(",").skip(P.optWhitespace), P.regex(/0?\.\d+|[01]/).map(Number), - (R, _, G, __, B, ___, A) => new LinearColorEntity$1({ + (R, _, G, __, B, ___, A) => new LinearColorEntity({ R: R / 255, G: G / 255, B: B / 255, @@ -1516,19 +1543,6 @@ class Grammar { ) } -class SerializerFactory { - - static #serializers = new Map() - - static registerSerializer(entity, object) { - SerializerFactory.#serializers.set(entity, object); - } - - static getSerializer(entity) { - return SerializerFactory.#serializers.get(Utility.getType(entity)) - } -} - /** * @template {IEntity} T */ @@ -1558,13 +1572,8 @@ class ISerializer { * @param {Boolean} insideString * @returns {String} */ - serialize(object, insideString) { - insideString ||= object.isShownAsString(); - let result = this.write(object, insideString); - if (object.isShownAsString()) { - result = `"${result}"`; - } - return result + serialize(object, insideString, entity = object) { + return this.write(entity, object, insideString) } /** @@ -1580,7 +1589,7 @@ class ISerializer { * @param {Boolean} insideString * @returns {String} */ - write(object, insideString) { + write(entity, object, insideString) { throw new Error("Not implemented") } @@ -1588,30 +1597,12 @@ class ISerializer { * @param {String[]} fullKey * @param {Boolean} insideString */ - writeValue(value, fullKey, insideString) { - if (value === null) { - return "()" - } - const serialize = v => SerializerFactory.getSerializer(Utility.getType(v)).serialize(v); - // This is an exact match (and not instanceof) to hit also primitive types (by accessing value.constructor they are converted to objects automatically) - switch (value?.constructor) { - case Function: - return this.writeValue(value(), fullKey, insideString) - case Boolean: - return Utility.firstCapital(value.toString()) - case Number: - return value.toString() - case String: - return insideString - ? `\\"${Utility.encodeString(value)}\\"` - : `"${Utility.encodeString(value)}"` - } - if (value instanceof Array) { - return `(${value.map(v => serialize(v) + ",").join("")})` - } - if (value instanceof IEntity) { - return serialize(value) + writeValue(entity, value, fullKey, insideString) { + const serializer = SerializerFactory.getSerializer(Utility.getType(value)); + if (!serializer) { + throw new Error("Unknown value type, a serializer must be registered in the SerializerFactory class") } + return serializer.write(entity, value, insideString) } /** @@ -1620,7 +1611,7 @@ class ISerializer { * @param {Boolean} insideString * @returns {String} */ - subWrite(key, object, insideString) { + subWrite(entity, key, object, insideString) { let result = ""; let fullKey = key.concat(""); const last = fullKey.length - 1; @@ -1630,13 +1621,18 @@ class ISerializer { if (value?.constructor === Object) { // Recursive call when finding an object result += (result.length ? this.separator : "") - + this.subWrite(fullKey, value, insideString); - } else if (value !== undefined && this.showProperty(object, fullKey, value)) { + + 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.separator : "") + this.prefix + this.attributeKeyPrinter(fullKey) + this.attributeValueConjunctionSign - + this.writeValue(value, fullKey, insideString); + + ( + isSerialized + ? `"${this.writeValue(entity, value, fullKey, true)}"` + : this.writeValue(entity, value, fullKey, insideString) + ); } } if (this.trailingSeparator && result.length && fullKey.length === 1) { @@ -1646,7 +1642,7 @@ class ISerializer { return result } - showProperty(object, attributeKey, attributeValue) { + showProperty(entity, object, attributeKey, attributeValue) { const attributes = this.entityType.attributes; const attribute = Utility.objectGet(attributes, attributeKey); if (attribute instanceof TypeInitialization) { @@ -1662,7 +1658,7 @@ class ObjectSerializer extends ISerializer { super(ObjectEntity, " ", "\n", false); } - showProperty(object, attributeKey, attributeValue) { + showProperty(entity, object, attributeKey, attributeValue) { switch (attributeKey.toString()) { case "Class": case "Name": @@ -1670,7 +1666,7 @@ class ObjectSerializer extends ISerializer { // Serielized separately return false } - return super.showProperty(object, attributeKey, attributeValue) + return super.showProperty(entity, object, attributeKey, attributeValue) } /** @@ -1699,9 +1695,9 @@ class ObjectSerializer extends ISerializer { * @param {ObjectEntity} object * @param {Boolean} insideString */ - write(object, insideString) { - let result = `Begin Object Class=${object.Class.path} Name=${this.writeValue(object.Name, ["Name"], insideString)} -${this.subWrite([], object, insideString) + write(entity, object, insideString) { + let result = `Begin Object Class=${object.Class.path} Name=${this.writeValue(entity, object.Name, ["Name"], insideString)} +${this.subWrite(entity, [], object, insideString) + object .CustomProperties.map(pin => this.separator @@ -3329,7 +3325,6 @@ class IInputPinTemplate extends PinTemplate { * @param {PinElement} pin */ getInputs(pin) { - Utility.decodeInputString(pin.entity.DefaultValue); return this.#inputContentElements.map(element => // Faster than innerText which causes reflow element.innerHTML.replaceAll("
", "\n")) @@ -3658,7 +3653,7 @@ class PinElement extends IElement { reflect: true, }, color: { - type: LinearColorEntity$1, + type: LinearColorEntity, converter: { fromAttribute: (value, type) => { return value ? ISerializer.grammar.LinearColorFromAnyColor.parse(value).value : null @@ -5274,8 +5269,8 @@ class GeneralSerializer extends ISerializer { * @param {Boolean} insideString * @returns {String} */ - write(object, insideString = false) { - let result = this.wrap(this.subWrite([], object, insideString)); + write(entity, object, insideString = false) { + let result = this.wrap(this.subWrite(entity, [], object, insideString)); return result } } @@ -5285,16 +5280,19 @@ class GeneralSerializer extends ISerializer { */ /** - * @template {IEntity} T + * @template {IEntity | Boolean | Number | String} T */ class CustomSerializer extends GeneralSerializer { + #objectWriter + /** + * @param {(v: T, insideString: Boolean) => String} objectWriter * @param {new () => T} entityType */ constructor(objectWriter, entityType) { super(undefined, entityType); - this.objectWriter = objectWriter; + this.#objectWriter = objectWriter; } /** @@ -5302,30 +5300,12 @@ class CustomSerializer extends GeneralSerializer { * @param {Boolean} insideString * @returns {String} */ - write(object, insideString = false) { - let result = this.objectWriter(object, insideString); + write(entity, object, insideString = false) { + let result = this.#objectWriter(object, insideString); return result } } -class PinSerializer extends GeneralSerializer { - - constructor() { - super(v => `${PinEntity.lookbehind} (${v})`, PinEntity, "", ",", true); - } - - /** - * @param {String[]} fullKey - * @param {Boolean} insideString - */ - writeValue(value, fullKey, insideString) { - if (value?.constructor === String && fullKey.length == 1 && fullKey[0] == "DefaultValue") { - return `"${Utility.encodeInputString(value)}"` - } - return super.writeValue(value, fullKey, insideString) - } -} - /** * @typedef {import("../entity/IEntity").default} IEntity */ @@ -5346,11 +5326,10 @@ class ToStringSerializer extends GeneralSerializer { * @param {T} object * @param {Boolean} insideString */ - write(object, insideString) { - let result = insideString || object.isShownAsString() - ? `"${object.toString().replaceAll('"', '\\"')}"` - : object.toString(); - return result + write(entity, object, insideString) { + return !insideString && object.constructor === String + ? `"${Utility.encodeString(object.toString())}"` // String will have quotes if not inside a string already + : Utility.encodeString(object.toString()) } } @@ -5358,6 +5337,44 @@ function initializeSerializerFactory() { const bracketsWrapped = v => `(${v})`; + SerializerFactory.registerSerializer( + null, + new CustomSerializer( + (nullValue, insideString) => "()", + null + ) + ); + + SerializerFactory.registerSerializer( + Array, + new CustomSerializer( + /** @param {Array} array */ + (array, insideString) => + `(${array + .map(v => + SerializerFactory.getSerializer(Utility.getType(v)).serialize(v, insideString) + "," + ) + .join("") + })`, + Array + ) + ); + + SerializerFactory.registerSerializer( + Boolean, + new CustomSerializer( + /** @param {Boolean} boolean */ + (boolean, insideString) => boolean + ? insideString + ? "true" + : "True" + : insideString + ? "false" + : "False", + Boolean + ) + ); + SerializerFactory.registerSerializer( FunctionReferenceEntity, new GeneralSerializer(bracketsWrapped, FunctionReferenceEntity) @@ -5380,8 +5397,8 @@ function initializeSerializerFactory() { ); SerializerFactory.registerSerializer( - LinearColorEntity$1, - new GeneralSerializer(bracketsWrapped, LinearColorEntity$1) + LinearColorEntity, + new GeneralSerializer(bracketsWrapped, LinearColorEntity) ); SerializerFactory.registerSerializer( @@ -5389,6 +5406,15 @@ function initializeSerializerFactory() { new GeneralSerializer(v => `${LocalizedTextEntity.lookbehind}(${v})`, LocalizedTextEntity, "", ", ", false, "", _ => "") ); + SerializerFactory.registerSerializer( + Number, + new CustomSerializer( + /** @param {Number} value */ + value => value.toString(), + Number + ) + ); + SerializerFactory.registerSerializer( ObjectEntity, new ObjectSerializer() @@ -5409,14 +5435,24 @@ function initializeSerializerFactory() { SerializerFactory.registerSerializer(PathSymbolEntity, new ToStringSerializer(PathSymbolEntity)); + SerializerFactory.registerSerializer( + PinEntity, + new GeneralSerializer(v => `${PinEntity.lookbehind} (${v})`, PinEntity, "", ",", true) + ); + SerializerFactory.registerSerializer( PinReferenceEntity, new GeneralSerializer(v => v, PinReferenceEntity, "", " ", false, "", _ => "") ); SerializerFactory.registerSerializer( - PinEntity, - new PinSerializer() + String, + new CustomSerializer( + (value, insideString) => insideString + ? Utility.encodeString(value) + : `"${Utility.encodeString(value)}"`, + String + ) ); } diff --git a/dist/ueblueprint.min.js b/dist/ueblueprint.min.js index c2b98c2..1f35324 100644 --- a/dist/ueblueprint.min.js +++ b/dist/ueblueprint.min.js @@ -14,10 +14,10 @@ const e=window.ShadowRoot&&(void 0===window.ShadyCSS||window.ShadyCSS.nativeShad * Copyright 2017 Google LLC * SPDX-License-Identifier: BSD-3-Clause */ -var g;p.finalized=!0,p.elementProperties=new Map,p.elementStyles=[],p.shadowRootOptions={mode:"open"},null==u||u({ReactiveElement:p}),(null!==(o=globalThis.reactiveElementVersions)&&void 0!==o?o:globalThis.reactiveElementVersions=[]).push("1.3.4");const m=globalThis.trustedTypes,f=m?m.createPolicy("lit-html",{createHTML:e=>e}):void 0,v=`lit$${(Math.random()+"").slice(9)}$`,b="?"+v,y=`<${b}>`,E=document,w=(e="")=>E.createComment(e),P=e=>null===e||"object"!=typeof e&&"function"!=typeof e,S=Array.isArray,k=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,x=/-->/g,C=/>/g,N=RegExp(">|[ \t\n\f\r](?:([^\\s\"'>=/]+)([ \t\n\f\r]*=[ \t\n\f\r]*(?:[^ \t\n\f\r\"'`<>=]|(\"|')|))|$)","g"),A=/'/g,$=/"/g,L=/^(?:script|style|textarea|title)$/i,O=(e=>(t,...n)=>({_$litType$:e,strings:t,values:n}))(1),D=Symbol.for("lit-noChange"),T=Symbol.for("lit-nothing"),M=new WeakMap,_=E.createTreeWalker(E,129,null,!1),I=(e,t)=>{const n=e.length-1,i=[];let r,s=2===t?"":"",o=k;for(let t=0;t"===l[0]?(o=null!=r?r:k,u=-1):void 0===l[1]?u=-2:(u=o.lastIndex-l[2].length,a=l[1],o=void 0===l[3]?N:'"'===l[3]?$:A):o===$||o===A?o=N:o===x||o===C?o=k:(o=N,r=void 0);const d=o===N&&e[t+1].startsWith("/>")?" ":"";s+=o===k?n+y:u>=0?(i.push(a),n.slice(0,u)+"$lit$"+n.slice(u)+v+d):n+v+(-2===u?(i.push(void 0),t):d)}const a=s+(e[n]||"")+(2===t?"":"");if(!Array.isArray(e)||!e.hasOwnProperty("raw"))throw Error("invalid template strings array");return[void 0!==f?f.createHTML(a):a,i]};class H{constructor({strings:e,_$litType$:t},n){let i;this.parts=[];let r=0,s=0;const o=e.length-1,a=this.parts,[l,u]=I(e,t);if(this.el=H.createElement(l,n),_.currentNode=this.el.content,2===t){const e=this.el.content,t=e.firstChild;t.remove(),e.append(...t.childNodes)}for(;null!==(i=_.nextNode())&&a.length0){i.textContent=m?m.emptyScript:"";for(let n=0;nS(e)||"function"==typeof(null==e?void 0:e[Symbol.iterator]))(e)?this.S(e):this.T(e)}j(e,t=this._$AB){return this._$AA.parentNode.insertBefore(e,t)}k(e){this._$AH!==e&&(this._$AR(),this._$AH=this.j(e))}T(e){this._$AH!==T&&P(this._$AH)?this._$AA.nextSibling.data=e:this.k(E.createTextNode(e)),this._$AH=e}$(e){var t;const{values:n,_$litType$:i}=e,r="number"==typeof i?this._$AC(e):(void 0===i.el&&(i.el=H.createElement(i.h,this.options)),i);if((null===(t=this._$AH)||void 0===t?void 0:t._$AD)===r)this._$AH.m(n);else{const e=new F(r,this),t=e.p(this.options);e.m(n),this.k(t),this._$AH=e}}_$AC(e){let t=M.get(e.strings);return void 0===t&&M.set(e.strings,t=new H(e)),t}S(e){S(this._$AH)||(this._$AH=[],this._$AR());const t=this._$AH;let n,i=0;for(const r of e)i===t.length?t.push(n=new B(this.j(w()),this.j(w()),this,this.options)):n=t[i],n._$AI(r),i++;i2||""!==n[0]||""!==n[1]?(this._$AH=Array(n.length-1).fill(new String),this.strings=n):this._$AH=T}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(e,t=this,n,i){const r=this.strings;let s=!1;if(void 0===r)e=j(this,e,t,0),s=!P(e)||e!==this._$AH&&e!==D,s&&(this._$AH=e);else{const i=e;let o,a;for(e=r[0],o=0;oe}):void 0,v=`lit$${(Math.random()+"").slice(9)}$`,b="?"+v,y=`<${b}>`,E=document,w=(e="")=>E.createComment(e),P=e=>null===e||"object"!=typeof e&&"function"!=typeof e,S=Array.isArray,k=/<(?:(!--|\/[^a-zA-Z])|(\/?[a-zA-Z][^>\s]*)|(\/?$))/g,x=/-->/g,C=/>/g,N=RegExp(">|[ \t\n\f\r](?:([^\\s\"'>=/]+)([ \t\n\f\r]*=[ \t\n\f\r]*(?:[^ \t\n\f\r\"'`<>=]|(\"|')|))|$)","g"),A=/'/g,$=/"/g,L=/^(?:script|style|textarea|title)$/i,O=(e=>(t,...n)=>({_$litType$:e,strings:t,values:n}))(1),T=Symbol.for("lit-noChange"),D=Symbol.for("lit-nothing"),M=new WeakMap,_=E.createTreeWalker(E,129,null,!1),H=(e,t)=>{const n=e.length-1,i=[];let r,s=2===t?"":"",o=k;for(let t=0;t"===l[0]?(o=null!=r?r:k,u=-1):void 0===l[1]?u=-2:(u=o.lastIndex-l[2].length,a=l[1],o=void 0===l[3]?N:'"'===l[3]?$:A):o===$||o===A?o=N:o===x||o===C?o=k:(o=N,r=void 0);const d=o===N&&e[t+1].startsWith("/>")?" ":"";s+=o===k?n+y:u>=0?(i.push(a),n.slice(0,u)+"$lit$"+n.slice(u)+v+d):n+v+(-2===u?(i.push(void 0),t):d)}const a=s+(e[n]||"")+(2===t?"":"");if(!Array.isArray(e)||!e.hasOwnProperty("raw"))throw Error("invalid template strings array");return[void 0!==f?f.createHTML(a):a,i]};class I{constructor({strings:e,_$litType$:t},n){let i;this.parts=[];let r=0,s=0;const o=e.length-1,a=this.parts,[l,u]=H(e,t);if(this.el=I.createElement(l,n),_.currentNode=this.el.content,2===t){const e=this.el.content,t=e.firstChild;t.remove(),e.append(...t.childNodes)}for(;null!==(i=_.nextNode())&&a.length0){i.textContent=m?m.emptyScript:"";for(let n=0;nS(e)||"function"==typeof(null==e?void 0:e[Symbol.iterator]))(e)?this.S(e):this.T(e)}j(e,t=this._$AB){return this._$AA.parentNode.insertBefore(e,t)}k(e){this._$AH!==e&&(this._$AR(),this._$AH=this.j(e))}T(e){this._$AH!==D&&P(this._$AH)?this._$AA.nextSibling.data=e:this.k(E.createTextNode(e)),this._$AH=e}$(e){var t;const{values:n,_$litType$:i}=e,r="number"==typeof i?this._$AC(e):(void 0===i.el&&(i.el=I.createElement(i.h,this.options)),i);if((null===(t=this._$AH)||void 0===t?void 0:t._$AD)===r)this._$AH.m(n);else{const e=new z(r,this),t=e.p(this.options);e.m(n),this.k(t),this._$AH=e}}_$AC(e){let t=M.get(e.strings);return void 0===t&&M.set(e.strings,t=new I(e)),t}S(e){S(this._$AH)||(this._$AH=[],this._$AR());const t=this._$AH;let n,i=0;for(const r of e)i===t.length?t.push(n=new B(this.j(w()),this.j(w()),this,this.options)):n=t[i],n._$AI(r),i++;i2||""!==n[0]||""!==n[1]?(this._$AH=Array(n.length-1).fill(new String),this.strings=n):this._$AH=D}get tagName(){return this.element.tagName}get _$AU(){return this._$AM._$AU}_$AI(e,t=this,n,i){const r=this.strings;let s=!1;if(void 0===r)e=j(this,e,t,0),s=!P(e)||e!==this._$AH&&e!==T,s&&(this._$AH=e);else{const i=e;let o,a;for(e=r[0],o=0;o{var i,r;const s=null!==(i=null==n?void 0:n.renderBefore)&&void 0!==i?i:t;let o=s._$litPart$;if(void 0===o){const e=null!==(r=null==n?void 0:n.renderBefore)&&void 0!==r?r:null;s._$litPart$=o=new B(t.insertBefore(w(),e),e,void 0,null!=n?n:{})}return o._$AI(e),o})(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 D}}q.finalized=!0,q._$litElement$=!0,null===(X=globalThis.litElementHydrateSupport)||void 0===X||X.call(globalThis,{LitElement:q});const Z=globalThis.litElementPolyfillSupport;null==Z||Z({LitElement:q}),(null!==(Y=globalThis.litElementVersions)&&void 0!==Y?Y:globalThis.litElementVersions=[]).push("3.2.2");class J{static deleteNodesKeyboardKey="Delete";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 linkCurveHeight=15;static linkCurveWidth=80;static linkMinWidth=100;static linkRightSVGPath=(e,t,n)=>{let i=100-e;return`M ${e} 0 C ${t} 0, ${n} 0, 50 50 S ${i-t+e} 100, ${i} 100`};static maxZoom=7;static minZoom=-12;static mouseWheelFactor=.2;static nodeDeleteEventName="ueb-node-delete";static nodeDragEventName="ueb-node-drag";static nodeDragLocalEventName="ueb-node-drag-local";static nodeName=(e,t)=>`${e}_${t}`;static nodeRadius=8;static nodeReflowEventName="ueb-node-reflow";static pinColor={bool:r``,default:r``,exec:r``,name:r``,real:r``,string:r``,"/Script/CoreUObject.LinearColor":r``};static selectAllKeyboardKey="(bCtrl=True,Key=A)";static trackingMouseEventName={begin:"ueb-tracking-mouse-begin",end:"ueb-tracking-mouse-end"};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 Q{#e;get target(){return this.#e}#t;get blueprint(){return this.#t}options;constructor(e,t,n){this.#e=e,this.#t=t,n.consumeEvent??=!1,n.listenOnFocus??=!1,n.unlistenOnTextEdit??=!1,this.options=n;let i=this;this.listenHandler=e=>i.listenEvents(),this.unlistenHandler=e=>i.unlistenEvents(),this.options.listenOnFocus&&(this.blueprint.addEventListener(J.focusEventName.begin,this.listenHandler),this.blueprint.addEventListener(J.focusEventName.end,this.unlistenHandler)),this.options.unlistenOnTextEdit&&(this.blueprint.addEventListener(J.editTextEventName.begin,this.unlistenHandler),this.blueprint.addEventListener(J.editTextEventName.end,this.listenHandler))}unlistenDOMElement(){this.unlistenEvents(),this.blueprint.removeEventListener(J.focusEventName.begin,this.listenHandler),this.blueprint.removeEventListener(J.focusEventName.end,this.unlistenHandler),this.blueprint.removeEventListener(J.editTextEventName.begin,this.unlistenHandler),this.blueprint.removeEventListener(J.editTextEventName.end,this.listenHandler)}listenEvents(){}unlistenEvents(){}}class ee{#n;get types(){return this.#n}set types(e){this.#n=e}#i;get stringFallback(){return this.#i}set stringFallback(e){this.#i=e}constructor([...e],t=!0){this.#n=[...new Set([...e,...t?[String]:[]])],this.#i=t}}class te{#r;get type(){return this.#r}set type(e){this.#r=e}#s=!0;get showDefault(){return this.#s}set showDefault(e){this.#s=e}#o;get value(){return this.#o}set value(e){this.#o=e}static sanitize(e,t){return void 0===t&&(t=e?.constructor),t&&t!==ee&&!(e?.constructor===t||e instanceof t)&&(e=new t(e)),(e instanceof Boolean||e instanceof Number||e instanceof String)&&(e=e.valueOf()),e}constructor(e,t=!0,n){void 0===n&&(n=e instanceof Array?[]:e instanceof ee?"":te.sanitize(new e)),this.#s=t,this.#r=e}}class ne{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 clamp(e,t,n){return Math.min(Math.max(e,t),n)}static getScale(e){return Number(getComputedStyle(e).getPropertyValue("--ueb-scale"))}static minDecimals(e,t=1){const n=e*10**t;return Math.abs(n%1)>Number.EPSILON?e.toString():e.toFixed(t)}static convertLocation(e,t){const n=1/ne.getScale(t),i=t.getBoundingClientRect();return[Math.round((e[0]-i.x)*n),Math.round((e[1]-i.y)*n)]}static objectGet(e,t,n){if(void 0!==e){if(!(t instanceof Array))throw new TypeError("Expected keys to be an array.");return 0!=t.length&&t[0]in e&&void 0!==e[t[0]]?1==t.length?e[t[0]]:ne.objectGet(e[t[0]],t.slice(1),n):n}}static objectSet(e,t,n,i=!1,r=Object){if(!(t instanceof Array))throw new TypeError("Expected keys to be an array.");if(1==t.length){if(i||t[0]in e||void 0===e[t[0]])return e[t[0]]=n,!0}else if(t.length>0)return!i||e[t[0]]instanceof Object||(e[t[0]]=new r),ne.objectSet(e[t[0]],t.slice(1),n,i,r);return!1}static equals(e,t){return(e=te.sanitize(e))===(t=te.sanitize(t))||(e instanceof Array&&t instanceof Array?e.length==t.length&&!e.find(((e,n)=>!ne.equals(e,t[n]))):void 0)}static firstCapital(e){return e.charAt(0).toUpperCase()+e.substring(1)}static getType(e){let t=e?.constructor;switch(t){case te:return ne.getType(e.type);case Function:return e;default:return t}}static snapToGrid(e,t){return 1===t?e:[t*Math.round(e[0]/t),t*Math.round(e[1]/t)]}static mergeArrays(e=[],t=[]){let n=[];for(let i=0;i{t(this[e])}))}}})}return!0}unsubscribe(e,t){let n=this.#l.get(e);if(!n?.includes(t))return!1;if(n.splice(n.indexOf(t),1),0==n.length){const t=Symbol.for(e+"Storage"),n=Symbol.for(e+"ValInfo"),i=this[n][0];this[n][1],Object.defineProperty(i?Object.getPrototypeOf(this):this,e,Object.getOwnPropertyDescriptor(i?Object.getPrototypeOf(this):this,t)),delete this[n],delete this[t]}return!0}}{static attributes={};#u=!1;constructor(e){super();const t=(e,n,i,r="")=>{for(let s of ne.mergeArrays(Object.getOwnPropertyNames(n),Object.getOwnPropertyNames(i??{}))){let o=n[s];const a=ne.getType(o);if(s in n?null==o||o instanceof te&&!o.showDefault||s in i||console.warn(`${this.constructor.name} adds property ${r}${s} not defined in the serialized data`):console.warn(`Property ${r}${s} is not defined in ${this.constructor.name}.attributes`),a===Object){e[s]={},t(e[s],n[s],i[s],s+".");continue}const l=ne.objectGet(i,[s]);if(void 0===l){if(o instanceof ie&&(o=o.calculate(this),a=ne.getType(o)),o instanceof te){if(!o.showDefault){e[s]=void 0;continue}o=o.value}e[s]=o instanceof Array?[]:te.sanitize(o,a)}else e[s]=te.sanitize(l,a)}},n=this.constructor.attributes;e.constructor!==Object&&1==Object.getOwnPropertyNames(n).length&&(e={[Object.getOwnPropertyNames(n)[0]]:e}),t(this,n,e)}isShownAsString(){return this.#u}setShowAsString(e){this.#u=e}}class se extends re{static attributes={type:String,path:String}}class oe extends re{static attributes={MemberParent:se,MemberName:""}}class ae extends re{static attributes={value:String};static generateGuid(e=!0){let t=new Uint32Array(4);!0===e&&crypto.getRandomValues(t);let n="";return t.forEach((e=>{n+=("0".repeat(8)+e.toString(16).toUpperCase()).slice(-8)})),new ae({value:n})}valueOf(){return this.value}toString(){return this.value}}class le extends re{static attributes={value:String};static attributeConverter={fromAttribute:(e,t)=>new le(e),toAttribute:(e,t)=>e.toString()};valueOf(){return this.value}toString(){return this.value}}class ue extends re{static attributes={value:Number};constructor(e=0){super(e),this.value=Math.round(this.value)}valueOf(){return this.value}toString(){return this.value.toString()}}class ce extends re{static lookbehind="INVTEXT";static attributes={value:String}}class de extends re{static attributes={ActionName:"",bShift:!1,bCtrl:!1,bAlt:!1,bCmd:!1,Key:le};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)}}class he extends re{static attributes={R:Number,G:Number,B:Number,A:Number};toString(){return ne.printLinearColor(this)}}class pe extends re{static lookbehind="NSLOCTEXT";static attributes={namespace:String,key:String,value:String}}class ge extends re{static attributes={value:String};valueOf(){return this.value}toString(){return this.value}}class me extends re{static attributes={objectName:ge,pinGuid:ae}}class fe extends re{static lookbehind="Pin";static attributes={PinId:ae,PinName:"",PinFriendlyName:new te(pe,!1,null),PinToolTip:"",Direction:new te(String,!1,""),PinType:{PinCategory:"",PinSubCategory:"",PinSubCategoryObject:se,PinSubCategoryMemberReference:null,PinValueType:null,ContainerType:se,bIsReference:!1,bIsConst:!1,bIsWeakPointer:!1,bIsUObjectWrapper:!1,bSerializeAsSinglePrecisionFloat:!1},LinkedTo:new te([me],!1),DefaultValue:new te(new ee([he]),!1),AutogeneratedDefaultValue:new te(String,!1),DefaultObject:new te(se,!1,null),PersistentGuid:ae,bHidden:!1,bNotConnectable:!1,bDefaultValueIsReadOnly:!1,bDefaultValueIsIgnored:!1,bAdvancedView:!1,bOrphanedPin:!1};getDefaultValue(){return this.DefaultValue??""}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){this.LinkedTo;const n=this.LinkedTo?.find((n=>n.objectName==e&&n.pinGuid.valueOf()==t.PinId.valueOf()));return!n&&((this.LinkedTo??(this.LinkedTo=[])).push(new me({objectName:e,pinGuid:t.PinId})),!0)}unlinkFrom(e,t){const n=this.LinkedTo?.findIndex((n=>n.objectName==e&&n.pinGuid.valueOf()==t.PinId.valueOf()));return n>=0&&(1==this.LinkedTo.length?this.LinkedTo=void 0:this.LinkedTo.splice(n,1),!0)}getType(){return"struct"==this.PinType.PinCategory?this.PinType.PinSubCategoryObject.path:this.PinType.PinCategory}getSubCategory(){return this.PinType.PinSubCategoryObject.path}}class ve extends re{static attributes={MemberName:String,MemberGuid:ae,bSelfContext:!1}}class be extends re{static attributes={Class:se,Name:"",bIsPureFunc:new te(Boolean,!1,!1),VariableReference:new te(ve,!1,null),FunctionReference:new te(oe,!1,null),EventReference:new te(oe,!1,null),TargetType:new te(se,!1,null),NodePosX:ue,NodePosY:ue,AdvancedPinDisplay:new te(le,!1,null),EnabledState:new te(le,!1,null),NodeGuid:ae,ErrorType:new te(ue,!1),ErrorMsg:new te(String,!1,""),CustomProperties:[fe]};static nameRegex=/(\w+)_(\d+)/;getObjectName(e=!1){return e?this.getNameAndCounter()[0]:this.Name}getNameAndCounter(){const e=this.getObjectName(!1).match(be.nameRegex);return e&&3==e.length?[e[1],parseInt(e[2])]:["",0]}getDisplayName(){let e=this.FunctionReference?.MemberName;return e?(e=ne.formatStringName(e),e):(e=ne.formatStringName(this.getNameAndCounter()[0]),e)}getCounter(){return this.getNameAndCounter()[1]}}"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self&&self;function ye(e){return e&&e.__esModule&&Object.prototype.hasOwnProperty.call(e,"default")?e.default:e}var Ee={exports:{}};"undefined"!=typeof self&&self;var we=ye(Ee.exports=function(e){var t={};function n(i){if(t[i])return t[i].exports;var r=t[i]={i:i,l:!1,exports:{}};return e[i].call(r.exports,r,r.exports,n),r.l=!0,r.exports}return n.m=e,n.c=t,n.d=function(e,t,i){n.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:i})},n.r=function(e){Object.defineProperty(e,"__esModule",{value:!0})},n.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return n.d(t,"a",t),t},n.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},n.p="",n(n.s=0)}([function(e,t,n){function i(e){if(!(this instanceof i))return new i(e);this._=e}var r=i.prototype;function s(e,t){for(var n=0;n>7),buf:function(e){var t=o((function(e,t,n,i){return e.concat(n===i.length-1?Buffer.from([t,0]).readUInt16BE(0):i.readUInt16BE(n))}),[],e);return Buffer.from(a((function(e){return(e<<1&65535)>>8}),t))}(n.buf)}})),n}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 d(e){c();var t=o((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 n,r=t/8,s=(n=function(e){return e>48},o((function(e,t){return e||(n(t)?t:e)}),null,e));if(s)throw new Error(s+" bit range requested exceeds 48 bit (6 byte) Number max.");return new i((function(t,n){var i=r+n;return i>t.length?P(n,r.toString()+" bytes"):w(i,o((function(e,t){var n=l(t,e.buf);return{coll:e.coll.concat(n.v),buf:n.buf}}),{coll:[],buf:t.slice(n,i)},e).coll)}))}function h(e,t){return new i((function(n,i){return c(),i+t>n.length?P(i,t+" bytes for "+e):w(i+t,n.slice(i,i+t))}))}function p(e,t){if("number"!=typeof(n=t)||Math.floor(n)!==n||t<0||t>6)throw new Error(e+" requires integer length in range [0, 6].");var n}function g(e){return p("uintBE",e),h("uintBE("+e+")",e).map((function(t){return t.readUIntBE(0,e)}))}function m(e){return p("uintLE",e),h("uintLE("+e+")",e).map((function(t){return t.readUIntLE(0,e)}))}function f(e){return p("intBE",e),h("intBE("+e+")",e).map((function(t){return t.readIntBE(0,e)}))}function v(e){return p("intLE",e),h("intLE("+e+")",e).map((function(t){return t.readIntLE(0,e)}))}function b(e){return e instanceof i}function y(e){return"[object Array]"==={}.toString.call(e)}function E(e){return u()&&Buffer.isBuffer(e)}function w(e,t){return{status:!0,index:e,value:t,furthest:-1,expected:[]}}function P(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 n=e.furthest===t.furthest?function(e,t){if(function(){if(void 0!==i._supportsSet)return i._supportsSet;var e="undefined"!=typeof Set;return i._supportsSet=e,e}()&&Array.from){for(var n=new Set(e),r=0;r=0;){if(o in n){i=n[o].line,0===s&&(s=n[o].lineStart);break}("\n"===e.charAt(o)||"\r"===e.charAt(o)&&"\n"!==e.charAt(o+1))&&(r++,0===s&&(s=o+1)),o--}var a=i+r,l=t-s;return n[t]={line:a,lineStart:s},{offset:t,line:a+1,column:l+1}}function C(e){if(!b(e))throw new Error("not a parser: "+e)}function N(e,t){return"string"==typeof e?e.charAt(t):e[t]}function A(e){if("number"!=typeof e)throw new Error("not a number: "+e)}function $(e){if("function"!=typeof e)throw new Error("not a function: "+e)}function L(e){if("string"!=typeof e)throw new Error("not a string: "+e)}var O=2,D=3,T=8,M=5*T,_=4*T,I=" ";function H(e,t){return new Array(t+1).join(e)}function j(e,t,n){var i=t-e.length;return i<=0?e:H(n,i)+e}function F(e,t,n,i){return{from:e-t>0?e-t:0,to:e+n>i?i:e+n}}function B(e,t){var n,i,r,s,l,u=t.index,c=u.offset,d=1;if(c===e.length)return"Got the end of the input";if(E(e)){var h=c-c%T,p=c-h,g=F(h,M,_+T,e.length),m=a((function(e){return a((function(e){return j(e.toString(16),2,"0")}),e)}),function(e,t){var n=e.length,i=[],r=0;if(n<=t)return[e.slice()];for(var s=0;s=4&&(n+=1),d=2,r=a((function(e){return e.length<=4?e.join(" "):e.slice(0,4).join(" ")+" "+e.slice(4).join(" ")}),m),(l=(8*(s.to>0?s.to-1:s.to)).toString(16).length)<2&&(l=2)}else{var f=e.split(/\r\n|[\n\r\u2028\u2029]/);n=u.column-1,i=u.line-1,s=F(i,O,D,f.length),r=f.slice(s.from,s.to),l=s.to.toString().length}var v=i-s.from;return E(e)&&(l=(8*(s.to>0?s.to-1:s.to)).toString(16).length)<2&&(l=2),o((function(t,i,r){var o,a=r===v,u=a?"> ":I;return o=E(e)?j((8*(s.from+r)).toString(16),l,"0"):j((s.from+r+1).toString(),l," "),[].concat(t,[u+o+" | "+i],a?[I+H(" ",l)+" | "+j("",n," ")+H("^",d)]:[])}),[],r).join("\n")}function z(e,t){return["\n","-- PARSING FAILED "+H("-",50),"\n\n",B(e,t),"\n\n",(n=t.expected,1===n.length?"Expected:\n\n"+n[0]:"Expected one of the following: \n\n"+n.join(", ")),"\n"].join("");var n}function G(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 R(){for(var e=[].slice.call(arguments),t=e.length,n=0;n=2?A(t):t=0;var n=function(e){return RegExp("^(?:"+e.source+")",G(e))}(e),r=""+e;return i((function(e,i){var s=n.exec(e.slice(i));if(s){if(0<=t&&t<=s.length){var o=s[0],a=s[t];return w(i+o.length,a)}return P(i,"valid match group (0 to "+s.length+") in "+r)}return P(i,r)}))}function q(e){return i((function(t,n){return w(n,e)}))}function Z(e){return i((function(t,n){return P(n,e)}))}function J(e){if(b(e))return i((function(t,n){var i=e._(t,n);return i.index=n,i.value="",i}));if("string"==typeof e)return J(X(e));if(e instanceof RegExp)return J(Y(e));throw new Error("not a string, regexp, or parser: "+e)}function Q(e){return C(e),i((function(t,n){var i=e._(t,n),r=t.slice(n,i.index);return i.status?P(n,'not "'+r+'"'):w(n,null)}))}function ee(e){return $(e),i((function(t,n){var i=N(t,n);return n=e.length?P(t,"any character/byte"):w(t+1,N(e,t))})),se=i((function(e,t){return w(e.length,e.slice(t))})),oe=i((function(e,t){return t=0})).desc(t)},i.optWhitespace=de,i.Parser=i,i.range=function(e,t){return ee((function(n){return e<=n&&n<=t})).desc(e+"-"+t)},i.regex=Y,i.regexp=Y,i.sepBy=K,i.sepBy1=W,i.seq=R,i.seqMap=U,i.seqObj=function(){for(var e,t={},n=0,r=(e=arguments,Array.prototype.slice.call(e)),s=r.length,o=0;o255)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 i((function(n,i){var r=N(n,i);return r===e?w(i+1,r):P(i,t)}))},buffer:function(e){return h("buffer",e).map((function(e){return Buffer.from(e)}))},encodedString:function(e,t){return h("string",t).map((function(t){return t.toString(e)}))},uintBE:g,uint8BE:g(1),uint16BE:g(2),uint32BE:g(4),uintLE:m,uint8LE:m(1),uint16LE:m(2),uint32LE:m(4),intBE:f,int8BE:f(1),int16BE:f(2),int32BE:f(4),intLE:v,int8LE:v(1),int16LE:v(2),int32LE:v(4),floatBE:h("floatBE",4).map((function(e){return e.readFloatBE(0)})),floatLE:h("floatLE",4).map((function(e){return e.readFloatLE(0)})),doubleBE:h("doubleBE",8).map((function(e){return e.readDoubleBE(0)})),doubleLE:h("doubleLE",8).map((function(e){return e.readDoubleLE(0)}))},e.exports=i}]));class Pe extends re{static attributes={X:Number,Y:Number,Z:Number}}let Se=we;class ke{static getGrammarForType(e,t,n){if(t instanceof te)return t=t.type,ke.getGrammarForType(e,t,n);if(t instanceof ee){const n=t.types.filter((e=>e!==String));let i=Se.alt(...n.map((t=>ke.getGrammarForType(e,t).wrap(Se.string('"'),Se.string('"')).map((e=>(e.setShowAsString(!0),e))))));return n.lengthke.getGrammarForType(e,ne.getType(t)))).reduce(((t,n)=>n&&t!==e.AttributeAnyValue?t.or(n):e.AttributeAnyValue)).trim(Se.optWhitespace).sepBy(Se.string(",")).skip(Se.regex(/,?\s*/)),Se.string(")"),((e,t,n)=>t));default:return n}}static createPropertyGrammar=(e,t,n=Se.string("=").trim(Se.optWhitespace))=>e.AttributeName.skip(n).chain((n=>{const i=n.split("."),r=ne.objectGet(t.attributes,i);return ke.getGrammarForType(e,r,e.AttributeAnyValue).map((e=>t=>ne.objectSet(t,i,e,!0)))}));static createEntityGrammar=(e,t)=>Se.seqMap(t.lookbehind?Se.seq(Se.string(t.lookbehind),Se.optWhitespace,Se.string("(")):Se.string("("),ke.createPropertyGrammar(e,t).trim(Se.optWhitespace).sepBy(Se.string(",")).skip(Se.regex(/,?/).then(Se.optWhitespace)),Se.string(")"),((e,n,i)=>{let r={};return n.forEach((e=>e(r))),new t(r)}));InlineWhitespace=e=>Se.regex(/[^\S\n]+/).desc("inline whitespace");InlineOptWhitespace=e=>Se.regex(/[^\S\n]*/).desc("inline optional whitespace");MultilineWhitespace=e=>Se.regex(/[^\S\n]*\n\s*/).desc("whitespace with at least a newline");Null=e=>Se.seq(Se.string("("),e.InlineOptWhitespace,Se.string(")")).map((e=>null)).desc("null: ()");Boolean=e=>Se.alt(Se.string("True"),Se.string("False")).map((e=>"True"===e)).desc("either True or False");HexDigit=e=>Se.regex(/[0-9a-fA-f]/).desc("hexadecimal digit");Number=e=>Se.regex(/[\-\+]?[0-9]+(?:\.[0-9]+)?/).map(Number).desc("a number");NaturalNumber=e=>Se.regex(/0|[1-9]\d*/).map(Number).desc("a natural number");ColorNumber=e=>e.NaturalNumber.assert((e=>0<=e&&e<256),"the color must be between 0 and 256 excluded");Word=e=>Se.regex(/[a-zA-Z]+/).desc("a word");String=e=>Se.regex(/(?:[^"\\]|\\.)*/).wrap(Se.string('"'),Se.string('"')).map(ne.decodeString).desc('string (with possibility to escape the quote using ")');ReferencePath=e=>Se.seq(Se.string("/"),e.PathSymbol.map((e=>e.toString())).sepBy1(Se.string(".")).tieWith(".")).tie().atLeast(2).tie().desc('a path (words with possibly underscore, separated by ".", separated by "/")');AttributeName=e=>e.Word.sepBy1(Se.string(".")).tieWith(".").desc('words separated by ""');None=e=>Se.string("None").map((e=>new se({type:"None",path:""}))).desc("none");Integer=e=>Se.regex(/[\-\+]?[0-9]+/).map((e=>new ue(e))).desc("an integer");Guid=e=>e.HexDigit.times(32).tie().map((e=>new ae({value:e}))).desc("32 digit hexadecimal value");Identifier=e=>Se.regex(/\w+/).map((e=>new le(e)));PathSymbol=e=>Se.regex(/[0-9\w]+/).map((e=>new ge({value:e})));Reference=e=>Se.alt(e.None,...[e.ReferencePath.map((e=>new se({type:"",path:e})))].flatMap((e=>[e,e.trim(Se.string('"'))])),Se.seqMap(e.Word,Se.optWhitespace,Se.alt(...[e.ReferencePath].flatMap((e=>[e.wrap(Se.string('"'),Se.string('"')),e.wrap(Se.string("'\""),Se.string("\"'"))]))),((e,t,n)=>new se({type:e,path:n}))),e.Word.map((e=>new se({type:e,path:""}))));LocalizedText=e=>Se.seqMap(Se.string(pe.lookbehind).skip(Se.optWhitespace).skip(Se.string("(")),e.String.trim(Se.optWhitespace),Se.string(","),e.String.trim(Se.optWhitespace),Se.string(","),e.String.trim(Se.optWhitespace),Se.string(")"),((e,t,n,i,r,s,o)=>new pe({namespace:t,key:i,value:s})));InvariantText=e=>e.String.trim(Se.optWhitespace).wrap(Se.string(ce.lookbehind).skip(Se.optWhitespace).skip(Se.string("(")),Se.string(")")).map((e=>new ce({value:e})));AttributeAnyValue=e=>Se.alt(e.Null,e.None,e.Boolean,e.Number,e.Integer,e.String,e.Guid,e.LocalizedText,e.InvariantText,e.Reference,e.Vector,e.LinearColor);PinReference=e=>Se.seqMap(e.PathSymbol,Se.whitespace,e.Guid,((e,t,n)=>new me({objectName:e,pinGuid:n})));Vector=e=>ke.createEntityGrammar(e,Pe);LinearColor=e=>ke.createEntityGrammar(e,he);FunctionReference=e=>ke.createEntityGrammar(e,oe);KeyBinding=e=>Se.alt(e.Identifier.map((e=>new de({Key:e}))),ke.createEntityGrammar(e,de));Pin=e=>ke.createEntityGrammar(e,fe);CustomProperties=e=>Se.string("CustomProperties").then(Se.whitespace).then(e.Pin).map((e=>t=>{let n=ne.objectGet(t,["CustomProperties"],[]);n.push(e),ne.objectSet(t,["CustomProperties"],n,!0)}));Object=e=>Se.seqMap(Se.seq(Se.string("Begin"),Se.whitespace,Se.string("Object"),Se.whitespace),Se.alt(e.CustomProperties,ke.createPropertyGrammar(e,be)).sepBy1(Se.whitespace),Se.seq(e.MultilineWhitespace,Se.string("End"),Se.whitespace,Se.string("Object")),((e,t,n)=>{let i={};return t.forEach((e=>e(i))),new be(i)}));MultipleObject=e=>e.Object.sepBy1(Se.whitespace).trim(Se.optWhitespace);LinearColorFromHex=e=>Se.string("#").then(e.HexDigit.times(2).tie().times(3,4)).trim(Se.optWhitespace).map((([e,t,n,i])=>new he({R:parseInt(e,16)/255,G:parseInt(t,16)/255,B:parseInt(n,16)/255,A:i?parseInt(i,16)/255:1})));LinearColorFromRGBList=e=>Se.seqMap(e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber.map(Number),((e,t,n,i,r)=>new he({R:e/255,G:n/255,B:r/255,A:1})));LinearColorFromRGB=e=>Se.string("rgb").then(e.LinearColorFromRGBList.wrap(Se.regex(/\(\s*/),Se.regex(/\s*\)/)));LinearColorFromRGBA=e=>Se.string("rgba").then(Se.seqMap(e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber.map(Number),Se.string(",").skip(Se.optWhitespace),Se.regex(/0?\.\d+|[01]/).map(Number),((e,t,n,i,r,s,o)=>new he({R:e/255,G:n/255,B:r/255,A:o}))).wrap(Se.regex(/\(\s*/),Se.regex(/\s*\)/)));LinearColorFromAnyColor=e=>Se.alt(e.LinearColorFromRGBList,e.LinearColorFromHex,e.LinearColorFromRGB,e.LinearColorFromRGBA)}class xe{static#c=new Map;static registerSerializer(e,t){xe.#c.set(e,t)}static getSerializer(e){return xe.#c.get(ne.getType(e))}}class Ce{static grammar=we.createLanguage(new ke);constructor(e,t,n,i,r,s){this.entityType=e,this.prefix=t??"",this.separator=n??",",this.trailingSeparator=i??!1,this.attributeValueConjunctionSign=r??"=",this.attributeKeyPrinter=s??(e=>e.join("."))}deserialize(e){return this.read(e)}serialize(e,t){t||=e.isShownAsString();let n=this.write(e,t);return e.isShownAsString()&&(n=`"${n}"`),n}read(e){throw new Error("Not implemented")}write(e,t){throw new Error("Not implemented")}writeValue(e,t,n){if(null===e)return"()";const i=e=>xe.getSerializer(ne.getType(e)).serialize(e);switch(e?.constructor){case Function:return this.writeValue(e(),t,n);case Boolean:return ne.firstCapital(e.toString());case Number:return e.toString();case String:return n?`\\"${ne.encodeString(e)}\\"`:`"${ne.encodeString(e)}"`}return e instanceof Array?`(${e.map((e=>i(e)+",")).join("")})`:e instanceof re?i(e):void 0}subWrite(e,t,n){let i="",r=e.concat("");const s=r.length-1;for(const e of Object.getOwnPropertyNames(t)){r[s]=e;const o=t[e];o?.constructor===Object?i+=(i.length?this.separator:"")+this.subWrite(r,o,n):void 0!==o&&this.showProperty(t,r,o)&&(i+=(i.length?this.separator:"")+this.prefix+this.attributeKeyPrinter(r)+this.attributeValueConjunctionSign+this.writeValue(o,r,n))}return this.trailingSeparator&&i.length&&1===r.length&&(i+=this.separator),i}showProperty(e,t,n){const i=this.entityType.attributes,r=ne.objectGet(i,t);return!(r instanceof te)||(!ne.equals(r.value,n)||r.showDefault)}}class Ne extends Ce{constructor(){super(be," ","\n",!1)}showProperty(e,t,n){switch(t.toString()){case"Class":case"Name":case"CustomProperties":return!1}return super.showProperty(e,t,n)}read(e){const t=Ce.grammar.Object.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}readMultiple(e){const t=Ce.grammar.MultipleObject.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}write(e,t){return`Begin Object Class=${e.Class.path} Name=${this.writeValue(e.Name,["Name"],t)}\n${this.subWrite([],e,t)+e.CustomProperties.map((e=>this.separator+this.prefix+"CustomProperties "+xe.getSerializer(fe).serialize(e))).join("")}\nEnd Object\n`}}class Ae extends Q{#d;constructor(e,t,n={}){n.listenOnFocus=!0,n.unlistenOnTextEdit=!0,super(e,t,n),this.serializer=new Ne;let i=this;this.#d=e=>i.copied()}listenEvents(){document.body.addEventListener("copy",this.#d)}unlistenEvents(){document.body.removeEventListener("copy",this.#d)}copied(){const e=this.blueprint.getNodes(!0).map((e=>this.serializer.serialize(e.entity,!1))).join("\n\n");navigator.clipboard.writeText(e)}}class $e{static styles=r``;#h=[];get inputObjects(){return this.#h}constructed(e){}connectedCallback(e){}willUpdate(e,t){}update(e,t){}render(e){return O``}firstUpdated(e,t){}updated(e,t){}inputSetup(e){this.#h=this.createInputObjects(e)}cleanup(e){this.#h.forEach((e=>e.unlistenDOMElement()))}createInputObjects(e){return[]}}class Le extends Q{#p;constructor(e,t,n={}){n.activateAnyKey??=!1,n.activationKeys??=[],n.listenOnFocus??=!0,n.unlistenOnTextEdit??=!0,n.activationKeys instanceof Array||(n.activationKeys=[n.activationKeys]),n.activationKeys=n.activationKeys.map((e=>{if(e instanceof de)return e;if(e.constructor===String){const t=Ce.grammar.KeyBinding.parse(e);if(t.status)return t.value}throw new Error("Unexpected key value")})),super(e,t,n),this.#p=this.options.activationKeys??[];let i=this;this.keyDownHandler=e=>{(this.options.activateAnyKey||i.#p.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&&J.Keys[t.Key]==e.code)))&&(n.consumeEvent&&e.stopImmediatePropagation(),i.fire(),document.removeEventListener("keydown",i.keyDownHandler),document.addEventListener("keyup",i.keyUpHandler))},this.keyUpHandler=e=>{(this.options.activateAnyKey||i.#p.some((t=>t.bShift&&"Shift"==e.key||t.bCtrl&&"Control"==e.key||t.bAlt&&"Alt"==e.key||t.bCmd&&"Meta"==e.key||J.Keys[t.Key]==e.code)))&&(n.consumeEvent&&e.stopImmediatePropagation(),i.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 Oe extends Le{constructor(e,t,n={}){n.activationKeys=J.deleteNodesKeyboardKey,super(e,t,n)}fire(){this.blueprint.removeGraphElement(...this.blueprint.getNodes(!0))}}class De extends Q{constructor(e,t,n){super(e,t,n),this.movementSpace=this.blueprint?.getGridDOMElement()??document.documentElement}locationFromEvent(e){const t=ne.convertLocation([e.clientX,e.clientY],this.movementSpace);return this.blueprint.compensateTranslation(t)}}class Te extends De{#g;#m;constructor(e,t,n){n.listenOnFocus=!0,super(e,t,n),this.looseTarget=n?.looseTarget??!0;let i=this;this.#g=e=>{e.preventDefault();const t=i.locationFromEvent(e);i.wheel(Math.sign(e.deltaY*J.mouseWheelFactor),t)},this.#m=e=>e.preventDefault(),this.blueprint.focused&&this.movementSpace.addEventListener("wheel",this.#g,!1)}listenEvents(){this.movementSpace.addEventListener("wheel",this.#g,!1),this.movementSpace.parentElement?.addEventListener("wheel",this.#m)}unlistenEvents(){this.movementSpace.removeEventListener("wheel",this.#g,!1),this.movementSpace.parentElement?.removeEventListener("wheel",this.#m)}wheel(e,t){}}class Me extends Te{#f=!1;get enableZoonIn(){return this.#f}set enableZoonIn(e){(e=Boolean(e))!=this.#f&&(this.#f=e)}wheel(e,t){let n=this.blueprint.getZoom();e=-e,!this.enableZoonIn&&0==n&&e>0||(n+=e,this.blueprint.setZoom(n,t))}}class _e extends Le{#v;constructor(e,t,n={}){n.activationKeys=J.enableZoomIn,super(e,t,n)}fire(){this.#v=this.blueprint.getInputObject(Me),this.#v.enableZoonIn=!0}unfire(){this.#v.enableZoonIn=!1}}class Ie extends Le{constructor(e,t,n={}){n.activationKeys=J.selectAllKeyboardKey,super(e,t,n)}fire(){this.blueprint.selectAll()}}class He extends De{#b;#y;#E;#w;#P=!1;started=!1;constructor(e,t,n={}){n.clickButton??=0,n.consumeEvent??=!0,n.exitAnyButton??=!0,n.looseTarget??=!1,n.moveEverywhere??=!1,super(e,t,n),this.clickedPosition=[0,0];const i=this.options.moveEverywhere?document.documentElement:this.movementSpace;let r=this;this.#b=e=>{if(r.blueprint.setFocused(!0),e.button===r.options.clickButton)(r.options.looseTarget||e.target==e.currentTarget)&&(r.options.consumeEvent&&e.stopImmediatePropagation(),i.addEventListener("mousemove",r.#y),document.addEventListener("mouseup",r.#w),r.clickedPosition=r.locationFromEvent(e),r.clicked(r.clickedPosition));else r.options.exitAnyButton||r.#w(e)},this.#y=e=>{r.options.consumeEvent&&e.stopImmediatePropagation(),i.removeEventListener("mousemove",r.#y),i.addEventListener("mousemove",r.#E);const t=r.getEvent(J.trackingMouseEventName.begin);r.#P=0==r.target.dispatchEvent(t);const n=r.locationFromEvent(e);r.startDrag(n),r.started=!0},this.#E=e=>{r.options.consumeEvent&&e.stopImmediatePropagation();const t=r.locationFromEvent(e),n=[e.movementX,e.movementY];r.dragTo(t,n),r.#P&&(r.blueprint.mousePosition=r.locationFromEvent(e))},this.#w=e=>{if(!r.options.exitAnyButton||e.button==r.options.clickButton){if(r.options.consumeEvent&&e.stopImmediatePropagation(),i.removeEventListener("mousemove",r.#y),i.removeEventListener("mousemove",r.#E),document.removeEventListener("mouseup",r.#w),r.started&&r.endDrag(),r.unclicked(),r.#P){const e=r.getEvent(J.trackingMouseEventName.end);r.target.dispatchEvent(e),r.#P=!1}r.started=!1}},this.listenEvents()}listenEvents(){this.target.addEventListener("mousedown",this.#b),2==this.options.clickButton&&this.target.addEventListener("contextmenu",(e=>e.preventDefault()))}unlistenEvents(){this.target.removeEventListener("mousedown",this.#b)}getEvent(e){return new CustomEvent(e,{detail:{tracker:this},bubbles:!0,cancelable:!0})}clicked(e){}startDrag(e){}dragTo(e,t){}endDrag(){}unclicked(e){}}class je extends He{startDrag(){this.blueprint.scrolling=!0}dragTo(e,t){this.blueprint.scrollDelta([-t[0],-t[1]])}endDrag(){this.blueprint.scrolling=!1}}class Fe extends De{#S=null;#k;#x;#C;constructor(e,t,n={}){n.listenOnFocus=!0,super(e,t,n);let i=this;this.#k=e=>{e.preventDefault(),i.blueprint.mousePosition=i.locationFromEvent(e)},this.#x=e=>{i.#S||(e.preventDefault(),this.#S=e.detail.tracker,i.unlistenMouseMove())},this.#C=e=>{i.#S==e.detail.tracker&&(e.preventDefault(),i.#S=null,i.listenMouseMove())}}listenMouseMove(){this.target.addEventListener("mousemove",this.#k)}unlistenMouseMove(){this.target.removeEventListener("mousemove",this.#k)}listenEvents(){this.listenMouseMove(),this.blueprint.addEventListener(J.trackingMouseEventName.begin,this.#x),this.blueprint.addEventListener(J.trackingMouseEventName.end,this.#C)}unlistenEvents(){this.unlistenMouseMove(),this.blueprint.removeEventListener(J.trackingMouseEventName.begin,this.#x),this.blueprint.removeEventListener(J.trackingMouseEventName.end,this.#C)}}class Be extends q{static properties={};#N=[];#t;get blueprint(){return this.#t}set blueprint(e){return this.#t=e}#A;get entity(){return this.#A}set entity(e){this.#A=e}#$;get template(){return this.#$}inputObjects=[];constructor(e,t){super(),this.#A=e,this.#$=t,this.inputObjects=[],this.#$.constructed(this)}createRenderRoot(){return this}connectedCallback(){super.connectedCallback(),this.blueprint=this.closest("ueb-blueprint"),this.template.connectedCallback(this)}willUpdate(e){super.willUpdate(e),this.template.willUpdate(this,e)}update(e){super.update(e),this.template.update(this,e)}render(){return this.template.render(this)}firstUpdated(e){super.firstUpdated(e),this.template.firstUpdated(this,e),this.template.inputSetup(this)}updated(e){super.updated(e),this.template.updated(this,e),this.#N.forEach((t=>t(e))),this.#N=[]}disconnectedCallback(){super.disconnectedCallback(),this.template.cleanup(this)}addNextUpdatedCallbacks(e,t=!1){this.#N.push(e),t&&this.requestUpdate()}isSameGraph(e){return this.blueprint&&this.blueprint==e?.blueprint}getInputObject(e){return this.template.inputObjects.find((t=>t.constructor==e))}}class ze extends Be{static properties={...super.properties,selected:{type:Boolean,attribute:"data-selected",reflect:!0,converter:ne.booleanConverter},locationX:{type:Number,attribute:!1},locationY:{type:Number,attribute:!1}};constructor(...e){super(...e),this.selected=!1,this.locationX=0,this.locationY=0,this.listeningDrag=!1;let t=this;this.dragHandler=e=>t.addLocation(e.detail.value)}connectedCallback(){super.connectedCallback(),this.setSelected(this.selected)}disconnectedCallback(){super.disconnectedCallback(),this.blueprint.removeEventListener(J.nodeDragEventName,this.dragHandler)}setLocation([e,t]){const n=[e-this.locationX,t-this.locationY];if(this.locationX=e,this.locationY=t,this.blueprint){const e=new CustomEvent(J.nodeDragLocalEventName,{detail:{value:n},bubbles:!1,cancelable:!0});this.dispatchEvent(e)}}addLocation([e,t]){this.setLocation([this.locationX+e,this.locationY+t])}setSelected(e=!0){this.selected=e,this.blueprint&&(this.selected?(this.listeningDrag=!0,this.blueprint.addEventListener(J.nodeDragEventName,this.dragHandler)):(this.blueprint.removeEventListener(J.nodeDragEventName,this.dragHandler),this.listeningDrag=!1))}dispatchDragEvent(e){const t=new CustomEvent(J.nodeDragEventName,{detail:{value:e},bubbles:!0,cancelable:!0});this.dispatchEvent(t)}snapToGrid(){const e=ne.snapToGrid([this.locationX,this.locationY],J.gridSize);this.locationX==e[0]&&this.locationY==e[1]||this.setLocation(e)}}class Ge extends He{constructor(e,t,n={}){n.consumeEvent=!0,super(e,t,n)}}class Re extends Be{static properties={...super.properties,initialPositionX:{type:Number,attribute:!1},initialPositionY:{type:Number,attribute:!1},finaPositionX:{type:Number,attribute:!1},finaPositionY:{type:Number,attribute:!1}};constructor(...e){super(...e),this.initialPositionX=0,this.initialPositionY=0,this.finaPositionX=0,this.finaPositionY=0}setBothLocations([e,t]){this.initialPositionX=e,this.initialPositionY=t,this.finaPositionX=e,this.finaPositionY=t}addSourceLocation([e,t]){this.initialPositionX+=e,this.initialPositionY+=t}addDestinationLocation([e,t]){this.finaPositionX+=e,this.finaPositionY+=t}}class Ue extends $e{update(e,t){super.update(e,t),t.has("initialPositionX")&&e.style.setProperty("--ueb-from-x",`${e.initialPositionX}`),t.has("initialPositionY")&&e.style.setProperty("--ueb-from-y",`${e.initialPositionY}`),t.has("finaPositionX")&&e.style.setProperty("--ueb-to-x",`${e.finaPositionX}`),t.has("finaPositionY")&&e.style.setProperty("--ueb-to-y",`${e.finaPositionY}`)}}class Ve extends Ue{static decreasingValue(e,t){const n=-e*t[0]**2,i=t[1]-n/t[0];return e=>n/e+i}static clampedLine(e,t){if(e[0]>t[0]){const n=e;e=t,t=n}const n=(t[1]-e[1])/(t[0]-e[0]),i=e[1]-n*e[0];return r=>rt[0]?t[1]:n*r+i}static c1DecreasingValue=Ve.decreasingValue(-.15,[100,15]);static c2DecreasingValue=Ve.decreasingValue(-.06,[500,130]);static c2Clamped=Ve.clampedLine([0,100],[200,30]);willUpdate(e,t){super.willUpdate(e,t);const n=Math.max(Math.abs(e.initialPositionX-e.finaPositionX),1),i=Math.max(n,J.linkMinWidth),r=n/i,s=e.originatesFromInput?e.initialPositionX ${""!=e.linkMessageIcon||""!=e.linkMessageText?O``:O``}`}}class Ke extends Re{static properties={...super.properties,source:{type:String,reflect:!0},destination:{type:String,reflect:!0},dragging:{type:Boolean,attribute:"data-dragging",converter:ne.booleanConverter,reflect:!0},originatesFromInput:{type:Boolean,attribute:!1},svgPathD:{type:String,attribute:!1},linkMessageIcon:{type:String,attribute:!1},linkMessageText:{type:String,attribute:!1}};#L;get sourcePin(){return this.#L}set sourcePin(e){this.#O(e,!1)}#D;get destinationPin(){return this.#D}set destinationPin(e){this.#O(e,!0)}#T;#M;#_;#I;#H;pathElement;constructor(e,t){super({},new Ve);const n=this;this.#T=()=>n.remove(),this.#M=e=>n.addSourceLocation(e.detail.value),this.#_=e=>n.addDestinationLocation(e.detail.value),this.#I=e=>n.setSourceLocation(),this.#H=e=>n.setDestinationLocation(),this.source=null,this.destination=null,this.dragging=!1,this.originatesFromInput=!1,this.startPercentage=0,this.svgPathD="",this.startPixels=0,this.linkMessageIcon="",this.linkMessageText="",e&&(this.sourcePin=e,t||(this.finaPositionX=this.initialPositionX,this.finaPositionY=this.initialPositionY)),t&&(this.destinationPin=t,e||(this.initialPositionX=this.finaPositionX,this.initialPositionY=this.finaPositionY)),this.#j()}#O(e,t){const n=()=>t?this.destinationPin:this.sourcePin;if(n()!=e){if(n()){const e=n().getNodeElement();e.removeEventListener(J.nodeDeleteEventName,this.#T),e.removeEventListener(J.nodeDragLocalEventName,t?this.#_:this.#M),e.removeEventListener(J.nodeReflowEventName,t?this.#H:this.#I),this.#F()}if(t?this.#D=e:this.#L=e,n()){const e=n().getNodeElement();e.addEventListener(J.nodeDeleteEventName,this.#T),e.addEventListener(J.nodeDragLocalEventName,t?this.#_:this.#M),e.addEventListener(J.nodeReflowEventName,t?this.#H:this.#I),t?this.setDestinationLocation():(this.setSourceLocation(),this.originatesFromInput=this.sourcePin.isInput()),this.#j()}}}#j(){this.sourcePin&&this.destinationPin&&(this.sourcePin.linkTo(this.destinationPin),this.destinationPin.linkTo(this.sourcePin))}#F(){this.sourcePin&&this.destinationPin&&(this.sourcePin.unlinkFrom(this.destinationPin),this.destinationPin.unlinkFrom(this.sourcePin))}disconnectedCallback(){super.disconnectedCallback(),this.#F(),this.sourcePin=null,this.destinationPin=null}setSourceLocation(e=null){if(null==e){const t=this;if(!this.hasUpdated||!this.sourcePin.hasUpdated)return void Promise.all([this.updateComplete,this.sourcePin.updateComplete]).then((()=>t.setSourceLocation()));e=this.sourcePin.template.getLinkLocation(this.sourcePin)}const[t,n]=e;this.initialPositionX=t,this.initialPositionY=n}setDestinationLocation(e=null){if(null==e){const t=this;if(!this.hasUpdated||!this.destinationPin.hasUpdated)return void Promise.all([this.updateComplete,this.destinationPin.updateComplete]).then((()=>t.setDestinationLocation()));e=this.destinationPin.template.getLinkLocation(this.destinationPin)}this.finaPositionX=e[0],this.finaPositionY=e[1]}startDragging(){this.dragging=!0}finishDragging(){this.dragging=!1}removeMessage(){this.linkMessageIcon="",this.linkMessageText=""}setMessageConvertType(){this.linkMessageIcon="ueb-icon-conver-type",this.linkMessageText=`Convert ${this.sourcePin.pinType} to ${this.destinationPin.pinType}.`}setMessageCorrect(){this.linkMessageIcon="ueb-icon-correct",this.linkMessageText=""}setMessageDirectionsIncompatible(){this.linkMessageIcon="ueb-icon-directions-incompatible",this.linkMessageText="Directions are not compatbile."}setMessagePlaceNode(){this.linkMessageIcon="ueb-icon-place-node",this.linkMessageText="Place a new node."}setMessageReplaceLink(){this.linkMessageIcon="ueb-icon-replace-link",this.linkMessageText="Replace existing input connections."}setMessageSameNode(){this.linkMessageIcon="ueb-icon-same-node",this.linkMessageText="Both are on the same node."}setMEssagetypesIncompatible(){this.linkMessageIcon="ueb-icon-types-incompatible",this.linkMessageText=`${this.sourcePin.pinType} is not compatible with ${this.destinationPin.pinType}.`}}customElements.define("ueb-link",Ke);class We extends He{#B;#z;#G;link;enteredPin;linkValid=!1;constructor(e,t,n){super(e,t,n);let i=this;this.#z=e=>{if(!i.enteredPin){i.linkValid=!1,i.enteredPin=e.target;const t=i.enteredPin,n=i.target;t.getNodeElement()==n.getNodeElement()?i.link.setMessageSameNode():t.isOutput()==n.isOutput()||t.isOutput()==n.isOutput()?i.link.setMessageDirectionsIncompatible():i.blueprint.getLinks([t,n]).length?(i.link.setMessageReplaceLink(),i.linkValid=!0):(i.link.setMessageCorrect(),i.linkValid=!0)}},this.#G=e=>{i.enteredPin==e.target&&(i.enteredPin=null,i.linkValid=!1,i.link?.setMessagePlaceNode())}}startDrag(e){this.link=new Ke(this.target,null),this.blueprint.linksContainerElement.prepend(this.link),this.link.setMessagePlaceNode(),this.#B=this.blueprint.querySelectorAll("ueb-pin"),this.#B.forEach((e=>{e!=this.target&&(e.getClickableElement().addEventListener("mouseenter",this.#z),e.getClickableElement().addEventListener("mouseleave",this.#G))})),this.link.startDragging(),this.link.setDestinationLocation(e)}dragTo(e,t){this.link.setDestinationLocation(e)}endDrag(){this.#B.forEach((e=>{e.removeEventListener("mouseenter",this.#z),e.removeEventListener("mouseleave",this.#G)})),this.enteredPin&&this.linkValid?(this.blueprint.addGraphElement(this.link),this.link.destinationPin=this.enteredPin,this.link.removeMessage(),this.link.finishDragging()):(this.link.finishDragging(),this.link.remove()),this.enteredPin=null,this.link=null,this.#B=null}}class Xe extends $e{static styles=r``;connectedCallback(e){super.connectedCallback(e),e.nodeElement=e.closest("ueb-node")}createInputObjects(e){return[new We(e.clickableElement,e.blueprint,{moveEverywhere:!0,looseTarget:!0})]}render(e){const t=O`
${this.renderIcon(e)}
`,n=O`
${e.getPinDisplayName()} ${this.renderInput(e)}
`;return O`
${e.isInput()?O`${t}${n}`:O`${n}${t}`}
`}renderIcon(e){return O``}renderInput(e){return O``}firstUpdated(e,t){super.firstUpdated(e,t),e.dataset.id=e.GetPinIdValue(),e.clickableElement=e}getLinkLocation(e){const t=e.querySelector(".ueb-pin-icon").getBoundingClientRect(),n=ne.convertLocation([(t.left+t.right)/2,(t.top+t.bottom)/2],e.blueprint.gridElement);return e.blueprint.compensateTranslation(n)}}class Ye extends Xe{#R;get inputContentElements(){return this.#R}firstUpdated(e,t){if(super.firstUpdated(e,t),this.#R=[...e.querySelectorAll(".ueb-pin-input-content")],this.#R.length){this.setInputs(e,this.getInputs(e));let t=this;this.onFocusHandler=t=>e.blueprint.dispatchEditTextEvent(!0),this.onFocusOutHandler=n=>{n.preventDefault(),document.getSelection()?.removeAllRanges(),t.setInputs(e,this.getInputs(e)),e.blueprint.dispatchEditTextEvent(!1)},this.#R.forEach((e=>{e.addEventListener("focus",this.onFocusHandler),e.addEventListener("focusout",this.onFocusOutHandler)}))}}cleanup(e){super.cleanup(e),this.#R.forEach((e=>{e.removeEventListener("focus",this.onFocusHandler),e.removeEventListener("focusout",this.onFocusOutHandler)}))}createInputObjects(e){return[...super.createInputObjects(e),...this.#R.map((t=>new Ge(t,e.blueprint)))]}getInput(e){return this.getInputs(e).reduce(((e,t)=>e+t),"")}getInputs(e){return ne.decodeInputString(e.entity.DefaultValue),this.#R.map((e=>e.innerHTML.replaceAll("
","\n")))}setInputs(e,t=[],n=!0){this.#R.forEach(((e,n)=>e.innerText=t[n])),n&&e.setDefaultValue(t.reduce(((e,t)=>e+t),""))}renderInput(e){return e.isInput()?O`
`:O``}}class qe extends Ye{#U;firstUpdated(e,t){super.firstUpdated(e,t),this.#U=e.querySelector(".ueb-pin-input");let n=this;this.onChangeHandler=t=>e.entity.DefaultValue=n.#U.checked?"true":"false",this.#U.addEventListener("change",this.onChangeHandler)}cleanup(e){super.cleanup(e),this.#U.removeEventListener("change",this.onChangeHandler)}getInputs(e){return[this.#U.checked?"true":"false"]}renderInput(e){return e.isInput()?O``:super.renderInput(e)}}class Ze extends Xe{renderIcon(e){return O``}}class Je extends Ye{#U;firstUpdated(e,t){super.firstUpdated(e,t),this.#U=e.querySelector(".ueb-pin-input")}getInputs(e){return[this.#U.dataset.linearColor]}setInputs(e,t=[]){}renderInput(e){return e.isInput()?O``:super.renderInput(e)}}class Qe extends Ye{onInputHandler;firstUpdated(e,t){super.firstUpdated(e,t),this.onInputHandler=e=>{e.stopPropagation(),("insertParagraph"==e.inputType||"insertLineBreak"==e.inputType||"insertFromPaste"==e.inputType&&e.target.innerText.includes("\n"))&&(e.target.blur(),this.inputContentElements.forEach((e=>e.innerText=e.innerText.replaceAll("\n",""))))},this.inputContentElements.forEach((e=>{e.addEventListener("input",this.onInputHandler)}))}cleanup(e){super.cleanup(e),this.inputContentElements.forEach((e=>{e.removeEventListener("input",this.onInputHandler)}))}getInputs(e){return this.inputContentElements.map((e=>e.textContent))}setInputs(e,t=[],n=!0){t=t.map((e=>e.replaceAll("\n",""))),super.setInputs(e,t,n)}}class et extends Ye{setInputs(e,t=[]){let n=parseFloat(t.length?t[0]:this.getInput(e)),i=!0;isNaN(n)&&(n=parseFloat(""!=e.entity.DefaultValue?e.entity.DefaultValue:e.entity.AutogeneratedDefaultValue)),isNaN(n)&&(n=0,i=!1),t[0]=ne.minDecimals(n),super.setInputs(e,t,i)}}class tt extends Ye{firstUpdated(e,t){super.firstUpdated(e,t)}}class nt extends Ye{#U;firstUpdated(e,t){super.firstUpdated(e,t),this.#U=e.querySelector(".ueb-pin-input")}getInputs(e){return[this.#U.dataset.linearColor]}setInputs(e,t=[]){}renderInput(e){return e.isInput()?O`
`:super.renderInput(e)}}class it extends Be{static#V={bool:qe,exec:Ze,name:Qe,real:et,string:tt,"/Script/CoreUObject.LinearColor":Je,"/Script/CoreUObject.Vector":nt};static properties={advancedView:{type:String,attribute:"data-advanced-view",reflect:!0},color:{type:he,converter:{fromAttribute:(e,t)=>e?Ce.grammar.LinearColorFromAnyColor.parse(e).value:null,toAttribute:(e,t)=>e?ne.printLinearColor(e):null},attribute:"data-color",reflect:!0},defaultValue:{type:String,attribute:!1},isLinked:{type:Boolean,converter:ne.booleanConverter,attribute:"data-linked",reflect:!0},pinType:{type:String,attribute:"data-type",reflect:!0},pinDirection:{type:String,attribute:"data-direction",reflect:!0}};static getTypeTemplate(e){return it.#V[e.getType()]??Xe}nodeElement;clickableElement;connections=0;get defaultValue(){return this.unreactiveDefaultValue}set defaultValue(e){let t=this.unreactiveDefaultValue;this.unreactiveDefaultValue=e,this.requestUpdate("defaultValue",t)}constructor(e){super(e,new(it.getTypeTemplate(e))),this.advancedView=e.bAdvancedView,this.unreactiveDefaultValue=e.getDefaultValue(),this.pinType=this.entity.getType(),this.color=this.constructor.properties.color.converter.fromAttribute(J.pinColor[this.pinType]?.toString()),this.isLinked=!1,this.pinDirection=e.isInput()?"input":e.isOutput()?"output":"hidden",this.entity.subscribe("DefaultValue",(e=>this.defaultValue=e.toString())),this.entity.subscribe("PinToolTip",(e=>{let t=e.match(/\s*(.+?(?=\n)|.+\S)\s*/);return t?ne.formatStringName(t[1]):ne.formatStringName(this.entity.PinName)}))}connectedCallback(){super.connectedCallback()}GetPinId(){return this.entity.PinId}GetPinIdValue(){return this.GetPinId().value}getPinName(){return this.entity.PinName}getPinDisplayName(){let e=null;return this.entity.PinToolTip&&(e=this.entity.PinToolTip.match(/\s*(.+?(?=\n)|.+\S)\s*/))?ne.formatStringName(e[1]):ne.formatStringName(this.entity.PinName)}isInput(){return this.entity.isInput()}isOutput(){return this.entity.isOutput()}getClickableElement(){return this.clickableElement}getLinkLocation(){return this.template.getLinkLocation(this)}getNodeElement(){return this.nodeElement}getLinks(){return this.entity.LinkedTo??[]}setDefaultValue(e){this.entity.DefaultValue=e}sanitizeLinks(){this.entity.LinkedTo=this.getLinks().filter((e=>{let t=this.blueprint.getPin(e);if(t){this.blueprint.getLink(this,t,!0)||this.blueprint.addGraphElement(new Ke(this,t))}return t}))}linkTo(e){this.entity.linkTo(e.getNodeElement().getNodeName(),e.entity),this.isLinked=this.entity.isLinked()}unlinkFrom(e){this.entity.unlinkFrom(e.getNodeElement().getNodeName(),e.entity),this.isLinked=this.entity.isLinked()}redirectLink(e,t){const n=this.entity.LinkedTo.findIndex((t=>t.objectName.toString()==e.getNodeElement().getNodeName()&&t.pinGuid.valueOf()==e.entity.PinId.valueOf()));return n>=0&&(this.entity.LinkedTo[n]=t,!0)}}customElements.define("ueb-pin",it);class rt extends He{constructor(e,t,n){super(e,t,n),this.stepSize=parseInt(n?.stepSize??J.gridSize),this.mouseLocation=[0,0]}startDrag(){this.mouseLocation=ne.snapToGrid(this.clickedPosition,this.stepSize),this.target.selected||(this.blueprint.unselectAll(),this.target.setSelected(!0))}dragTo(e,t){const n=[this.target.locationX,this.target.locationY],[i,r]=this.stepSize>1?[ne.snapToGrid(e,this.stepSize),ne.snapToGrid(n,this.stepSize)]:[e,n],s=[i[0]-this.mouseLocation[0],i[1]-this.mouseLocation[1]];0==s[0]&&0==s[1]||(s[0]+=r[0]-this.target.locationX,s[1]+=r[1]-this.target.locationY,this.target.dispatchDragEvent(s),this.mouseLocation=i)}unclicked(){this.started||(this.blueprint.unselectAll(),this.target.setSelected(!0))}}class st extends $e{createInputObjects(e){return[...super.createInputObjects(e),new rt(e,e.blueprint,{looseTarget:!0})]}update(e,t){super.update(e,t),t.has("locationX")&&e.style.setProperty("--ueb-position-x",`${e.locationX}`),t.has("locationY")&&e.style.setProperty("--ueb-position-y",`${e.locationY}`)}firstUpdated(e,t){super.firstUpdated(e,t),e.selected&&!e.listeningDrag&&e.setSelected(!0)}}class ot extends st{toggleAdvancedDisplayHandler;render(e){return O`
${e.nodeDisplayName}
${"DevelopmentOnly"==e.enabledState?.toString()?O`
Development Only
`:O``} ${e.advancedPinDisplay?O`
`:O``}
`}async firstUpdated(e,t){super.firstUpdated(e,t);const n=e.querySelector(".ueb-node-inputs"),i=e.querySelector(".ueb-node-outputs");Promise.all(e.getPinElements().map((e=>e.updateComplete))).then((()=>e.dispatchReflowEvent())),e.getPinElements().forEach((e=>{e.isInput()?n.appendChild(e):e.isOutput()&&i.appendChild(e)})),this.toggleAdvancedDisplayHandler=t=>{e.toggleShowAdvancedPinDisplay(),e.addNextUpdatedCallbacks((()=>e.dispatchReflowEvent()),!0)},e.nodeNameElement=e.querySelector(".ueb-node-name-text")}getPinElements(e){return e.querySelectorAll("ueb-pin")}}class at extends ze{static properties={...ze.properties,name:{type:String,attribute:"data-name",reflect:!0},advancedPinDisplay:{type:String,attribute:"data-advanced-display",converter:le.attributeConverter,reflect:!0},enabledState:{type:String,attribute:"data-enabled-state",reflect:!0},nodeDisplayName:{type:String,attribute:!1},pureFunction:{type:Boolean,converter:ne.booleanConverter,attribute:"data-pure-function"}};get blueprint(){return super.blueprint}set blueprint(e){super.blueprint=e,this.#K.forEach((t=>t.blueprint=e))}#W;get nodeNameElement(){return this.#W}set nodeNameElement(e){this.#W=e}#K;constructor(e){super(e,new ot),this.#K=this.getPinEntities().filter((e=>!e.isHidden())).map((e=>new it(e))),this.#K.forEach((e=>e.nodeElement=this)),this.name=e.getObjectName(),this.advancedPinDisplay=e.AdvancedPinDisplay?.toString(),this.enabledState=e.EnabledState,this.nodeDisplayName=e.getDisplayName(),this.pureFunction=e.bIsPureFunc,this.dragLinkObjects=[],super.setLocation([this.entity.NodePosX.value,this.entity.NodePosY.value]),this.entity.subscribe("AdvancedPinDisplay",(e=>this.advancedPinDisplay=e)),this.entity.subscribe("Name",(e=>this.name=e))}static fromSerializedObject(e){e=e.trim();let t=xe.getSerializer(be).deserialize(e);return new at(t)}connectedCallback(){this.getAttribute("type")?.trim(),super.connectedCallback()}disconnectedCallback(){super.disconnectedCallback(),this.dispatchDeleteEvent()}getNodeName(){return this.entity.getObjectName()}getNodeDisplayName(){return this.entity.getDisplayName()}sanitizeLinks(){this.getPinElements().forEach((e=>e.sanitizeLinks()))}rename(e){if(this.entity.Name==e)return!1;for(let t of this.getPinElements())for(let n of t.getLinks())this.blueprint.getPin(n).redirectLink(t,new me({objectName:e,pinGuid:t.entity.PinId}));this.entity.Name=e}getPinElements(){return this.#K}getPinEntities(){return this.entity.CustomProperties.filter((e=>e instanceof fe))}setLocation(e=[0,0]){let t=this.entity.NodePosX.constructor;this.entity.NodePosX=new t(e[0]),this.entity.NodePosY=new t(e[1]),super.setLocation(e)}dispatchDeleteEvent(e){let t=new CustomEvent(J.nodeDeleteEventName,{bubbles:!0,cancelable:!0});this.dispatchEvent(t)}dispatchReflowEvent(){let e=new CustomEvent(J.nodeReflowEventName,{bubbles:!0,cancelable:!0});this.dispatchEvent(e)}setShowAdvancedPinDisplay(e){this.entity.AdvancedPinDisplay=new le(e?"Shown":"Hidden")}toggleShowAdvancedPinDisplay(){this.setShowAdvancedPinDisplay("Shown"!=this.entity.AdvancedPinDisplay?.toString())}}customElements.define("ueb-node",at);class lt extends Q{#X;constructor(e,t,n={}){n.listenOnFocus=!0,n.unlistenOnTextEdit=!0,super(e,t,n),this.serializer=new Ne;let i=this;this.#X=e=>i.pasted(e.clipboardData.getData("Text"))}listenEvents(){document.body.addEventListener("paste",this.#X)}unlistenEvents(){document.body.removeEventListener("paste",this.#X)}pasted(e){let t=0,n=0,i=0,r=this.serializer.readMultiple(e).map((e=>{let r=new at(e);return t+=r.locationY,n+=r.locationX,++i,r}));t/=i,n/=i,r.length>0&&this.blueprint.unselectAll();let s=this.blueprint.mousePosition;return r.forEach((e=>{const i=[s[0]-n,s[1]-t];e.addLocation(i),e.snapToGrid(),e.setSelected(!0)})),this.blueprint.addGraphElement(...r),!0}}class ut extends He{constructor(e,t,n){super(e,t,n),this.selectorElement=this.blueprint.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 ct{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 dt{constructor(e,t,n,i){this.initialPosition=e,this.finalPosition=e,this.metadata=new Array(t.length),this.primaryOrder=new ct((e=>this.metadata[e].primaryBoundary)),this.secondaryOrder=new ct((e=>this.metadata[e].secondaryBoundary)),this.selectFunc=i,this.rectangles=t,this.primaryOrder.reserve(this.rectangles.length),this.secondaryOrder.reserve(this.rectangles.length),t.forEach(((e,t)=>{let r={primaryBoundary:this.initialPosition[0],secondaryBoundary:this.initialPosition[1],rectangle:t,onSecondaryAxis:!1};this.metadata[t]=r,i(e,!1);const s=n(e);this.initialPosition[1]{if(this.metadata[n].onSecondaryAxis)this.selectFunc(this.rectangles[n],i);else if(i){this.secondaryOrder.insert(n,e[1]);const i=this.metadata[n].secondaryBoundary;Math.sign(e[1]-i)==t[1]&&Math.sign(i-this.initialPosition[1])==t[1]&&this.selectFunc(this.rectangles[n],!0)}else this.selectFunc(this.rectangles[n],!1),this.secondaryOrder.remove(n);this.computeBoundaries(),this.selectTo(e)};e[0]this.boundaries.primaryN.v&&e[0]this.boundaries.primaryP.v&&(++this.primaryOrder.currentPosition,n(this.boundaries.primaryP.i,this.initialPosition[0]{this.selectFunc(this.rectangles[t],n),this.computeBoundaries(),this.selectTo(e)};e[1]this.boundaries.secondaryN.v&&e[1]this.boundaries.secondaryP.v&&(++this.secondaryOrder.currentPosition,i(this.boundaries.secondaryP.i,this.initialPosition[1]i.clickedSomewhere(e.target),this.blueprint.focus&&document.addEventListener("click",this.#Y)}clickedSomewhere(e){e.closest("ueb-blueprint")||this.blueprint.setFocused(!1)}listenEvents(){document.addEventListener("click",this.#Y)}unlistenEvents(){document.removeEventListener("click",this.#Y)}}class mt extends $e{static styleVariables={"--ueb-font-size":`${J.fontSize}`,"--ueb-grid-axis-line-color":`${J.gridAxisLineColor}`,"--ueb-grid-expand":`${J.expandGridSize}px`,"--ueb-grid-line-color":`${J.gridLineColor}`,"--ueb-grid-line-width":`${J.gridLineWidth}px`,"--ueb-grid-set-line-color":`${J.gridSetLineColor}`,"--ueb-grid-set":`${J.gridSet}`,"--ueb-grid-size":`${J.gridSize}px`,"--ueb-link-min-width":`${J.linkMinWidth}`,"--ueb-node-radius":`${J.nodeRadius}px`,"--ueb-pin-bool-color":`${J.pinColor.bool}`,"--ueb-pin-default-color":`${J.pinColor.default}`,"--ueb-pin-exec-color":`${J.pinColor.exec}`,"--ueb-pin-name-color":`${J.pinColor.name}`,"--ueb-pin-real-color":`${J.pinColor.real}`,"--ueb-pin-string-color":`${J.pinColor.string}`,"--ueb-pin-linear-color":`${J.pinColor["/Script/CoreUObject.LinearColor"]}`};constructed(e){e.style.cssText=Object.entries(mt.styleVariables).map((([e,t])=>`${e}:${t};`)).join("")}createInputObjects(e){return[new Ae(e.getGridDOMElement(),e),new lt(e.getGridDOMElement(),e),new Oe(e.getGridDOMElement(),e),new Ie(e.getGridDOMElement(),e),new Me(e.getGridDOMElement(),e,{looseTarget:!0}),new ut(e.getGridDOMElement(),e,{clickButton:0,exitAnyButton:!0,looseTarget:!0,moveEverywhere:!0}),new je(e.getGridDOMElement(),e,{clickButton:2,exitAnyButton:!1,looseTarget:!0,moveEverywhere:!0}),new gt(e.getGridDOMElement(),e),new Fe(e.getGridDOMElement(),e),new _e(e.getGridDOMElement(),e)]}render(e){return O`
1:1
`}firstUpdated(e,t){super.firstUpdated(e,t),e.headerElement=e.querySelector(".ueb-viewport-header"),e.overlayElement=e.querySelector(".ueb-viewport-overlay"),e.viewportElement=e.querySelector(".ueb-viewport-body"),e.selectorElement=new pt,e.querySelector(".ueb-grid-content")?.append(e.selectorElement),e.gridElement=e.viewportElement.querySelector(".ueb-grid"),e.linksContainerElement=e.querySelector("[data-links]"),e.linksContainerElement.append(...e.getLinks()),e.nodesContainerElement=e.querySelector("[data-nodes]"),e.nodesContainerElement.append(...e.getNodes()),e.viewportElement.scroll(J.expandGridSize,J.expandGridSize)}updated(e,t){super.updated(e,t),(t.has("scrollX")||t.has("scrollY"))&&e.viewportElement.scroll(e.scrollX,e.scrollY)}getPin(e,t){return e.querySelector(`ueb-node[data-name="${t.objectName}"] ueb-pin[data-id="${t.pinGuid}"]`)}}class ft extends Be{static properties={selecting:{type:Boolean,attribute:"data-selecting",reflect:!0,converter:ne.booleanConverter},scrolling:{type:Boolean,attribute:"data-scrolling",reflect:!0,converter:ne.booleanConverter},focused:{type:Boolean,attribute:"data-focused",reflect:!0,converter:ne.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 styles=mt.styles;#q=new Map;nodes=[];links=[];mousePosition=[0,0];gridElement;viewportElement;overlayElement;selectorElement;linksContainerElement;nodesContainerElement;headerElement;focused=!1;nodeBoundariesSupplier=e=>{let t=e.getBoundingClientRect(),n=this.nodesContainerElement.getBoundingClientRect();const i=1/this.getScale();return{primaryInf:(t.left-n.left)*i,primarySup:(t.right-n.right)*i,secondaryInf:(t.top-n.top)*i,secondarySup:(t.bottom-n.bottom)*i}};nodeSelectToggleFunction=(e,t)=>{e.setSelected(t)};constructor(e=new J){super({},new mt),this.selecting=!1,this.scrolling=!1,this.focused=!1,this.zoom=0,this.scrollX=J.expandGridSize,this.scrollY=J.expandGridSize,this.translateX=J.expandGridSize,this.translateY=J.expandGridSize}getGridDOMElement(){return this.gridElement}disconnectedCallback(){super.disconnectedCallback()}getScroll(){return[this.scrollX,this.scrollY]}setScroll([e,t],n=!1){this.scrollX=e,this.scrollY=t}scrollDelta(e,t=!1){const n=[2*J.expandGridSize,2*J.expandGridSize];let i=this.getScroll(),r=[i[0]+e[0],i[1]+e[1]],s=[0,0];for(let t=0;t<2;++t)e[t]<0&&r[t]0&&r[t]>n[t]-J.gridExpandThreshold*J.expandGridSize&&(s[t]=1);0==s[0]&&0==s[1]||this.seamlessExpand(s),i=this.getScroll(),r=[i[0]+e[0],i[1]+e[1]],this.setScroll(r,t)}scrollCenter(){const e=this.getScroll(),t=[this.translateX-e[0],this.translateY-e[1]],n=this.getViewportSize().map((e=>e/2)),i=[t[0]-n[0],t[1]-n[1]];this.scrollDelta(i,!0)}getViewportSize(){return[this.viewportElement.clientWidth,this.viewportElement.clientHeight]}getScrollMax(){return[this.viewportElement.scrollWidth-this.viewportElement.clientWidth,this.viewportElement.scrollHeight-this.viewportElement.clientHeight]}snapToGrid(e){return ne.snapToGrid(e,J.gridSize)}seamlessExpand([e,t]){e=Math.round(e),t=Math.round(t);let n=this.getScale();[e,t]=[-e*J.expandGridSize,-t*J.expandGridSize],0!=e&&(this.scrollX+=e,e/=n),0!=t&&(this.scrollY+=t,t/=n),this.translateX+=e,this.translateY+=t}progressiveSnapToGrid(e){return J.expandGridSize*Math.round(e/J.expandGridSize+.5*Math.sign(e))}getZoom(){return this.zoom}setZoom(e,t){if((e=ne.clamp(e,J.minZoom,J.maxZoom))==this.zoom)return;let n=this.getScale();this.zoom=e,t&&requestAnimationFrame((e=>{t[0]+=this.translateX,t[1]+=this.translateY;let i=this.getScale()/n,r=[i*t[0],i*t[1]];this.scrollDelta([(r[0]-t[0])*n,(r[1]-t[1])*n])}))}getScale(){return parseFloat(getComputedStyle(this.gridElement).getPropertyValue("--ueb-scale"))}compensateTranslation([e,t]){return[e-=this.translateX,t-=this.translateY]}getNodes(e=!1){return e?this.nodes.filter((e=>e.selected)):this.nodes}getPin(e){return[...this.nodes.find((t=>e.objectName.toString()==t.getNodeName()))?.getPinElements()??[]].find((t=>e.pinGuid.toString()==t.GetPinIdValue()))}getLinks([e,t]=[]){if(null==e!=t==null){const n=e??t;return this.links.filter((e=>e.sourcePin==n||e.destinationPin==n))}return null!=e&&null!=t?this.links.filter((n=>n.sourcePin==e&&n.destinationPin==t||n.sourcePin==t&&n.destinationPin==e)):this.links}getLink(e,t,n=!1){return this.links.find((i=>i.sourcePin==e&&i.destinationPin==t||n&&i.sourcePin==t&&i.destinationPin==e))}selectAll(){this.getNodes().forEach((e=>this.nodeSelectToggleFunction(e,!0)))}unselectAll(){this.getNodes().forEach((e=>this.nodeSelectToggleFunction(e,!1)))}addGraphElement(...e){for(let t of e)if(t.blueprint=this,t instanceof at&&!this.nodes.includes(t)){const e=t.entity.getObjectName(),n=this.nodes.find((t=>t.entity.getObjectName()==e));if(n){let e=n.entity.getObjectName(!0);this.#q[e]=this.#q[e]??-1;do{++this.#q[e]}while(this.nodes.find((t=>t.entity.getObjectName()==J.nodeName(e,this.#q[e]))));n.rename(J.nodeName(e,this.#q[e]))}this.nodes.push(t),this.nodesContainerElement?.appendChild(t)}else t instanceof Ke&&!this.links.includes(t)&&(this.links.push(t),this.linksContainerElement&&!this.linksContainerElement.contains(t)&&this.linksContainerElement.appendChild(t));e.filter((e=>e instanceof at)).forEach((e=>e.sanitizeLinks()))}removeGraphElement(...e){for(let t of e)if(t.closest("ueb-blueprint")==this){t.remove();let e=t instanceof at?this.nodes:t instanceof Ke?this.links:null;e?.splice(e.findIndex((e=>e===t)),1)}}setFocused(e=!0){if(this.focused==e)return;let t=new CustomEvent(e?"blueprint-focus":"blueprint-unfocus");this.focused=e,this.focused||this.unselectAll(),this.dispatchEvent(t)}dispatchEditTextEvent(e){const t=new CustomEvent(e?J.editTextEventName.begin:J.editTextEventName.end);this.dispatchEvent(t)}}customElements.define("ueb-blueprint",ft);class vt extends Ce{constructor(e,t,n,i,r,s,o){e=e??(e=>`(${e})`),super(t,n,i,r,s,o),this.wrap=e}read(e){const t=ke.getGrammarForType(Ce.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=!1){return this.wrap(this.subWrite([],e,t))}}class bt extends vt{constructor(e,t){super(void 0,t),this.objectWriter=e}write(e,t=!1){return this.objectWriter(e,t)}}class yt extends vt{constructor(){super((e=>`${fe.lookbehind} (${e})`),fe,"",",",!0)}writeValue(e,t,n){return e?.constructor===String&&1==t.length&&"DefaultValue"==t[0]?`"${ne.encodeInputString(e)}"`:super.writeValue(e,t,n)}}class Et extends vt{constructor(e){super(void 0,e)}write(e,t){return t||e.isShownAsString()?`"${e.toString().replaceAll('"','\\"')}"`:e.toString()}}!function(){const e=e=>`(${e})`;xe.registerSerializer(oe,new vt(e,oe)),xe.registerSerializer(ae,new Et(ae)),xe.registerSerializer(le,new Et(le)),xe.registerSerializer(ue,new Et(ue)),xe.registerSerializer(ce,new vt((e=>`${ce.lookbehind}(${e})`),ce,"",", ",!1,"",(e=>""))),xe.registerSerializer(de,new vt(e,de)),xe.registerSerializer(he,new vt(e,he)),xe.registerSerializer(pe,new vt((e=>`${pe.lookbehind}(${e})`),pe,"",", ",!1,"",(e=>""))),xe.registerSerializer(be,new Ne),xe.registerSerializer(se,new bt((e=>(e.type??"")+(e.path?e.type?`'"${e.path}"'`:`"${e.path}"`:"")),se)),xe.registerSerializer(ge,new Et(ge)),xe.registerSerializer(me,new vt((e=>e),me,""," ",!1,"",(e=>""))),xe.registerSerializer(fe,new yt)}();export{ft as Blueprint,J as Configuration,Ke as LinkElement,at as NodeElement}; +var X,Y;null==W||W(I,B),(null!==(g=globalThis.litHtmlVersions)&&void 0!==g?g:globalThis.litHtmlVersions=[]).push("2.2.7");class q extends p{constructor(){super(...arguments),this.renderOptions={host:this},this._$Do=void 0}createRenderRoot(){var e,t;const n=super.createRenderRoot();return null!==(e=(t=this.renderOptions).renderBefore)&&void 0!==e||(t.renderBefore=n.firstChild),n}update(e){const t=this.render();this.hasUpdated||(this.renderOptions.isConnected=this.isConnected),super.update(e),this._$Do=((e,t,n)=>{var i,r;const s=null!==(i=null==n?void 0:n.renderBefore)&&void 0!==i?i:t;let o=s._$litPart$;if(void 0===o){const e=null!==(r=null==n?void 0:n.renderBefore)&&void 0!==r?r:null;s._$litPart$=o=new B(t.insertBefore(w(),e),e,void 0,null!=n?n:{})}return o._$AI(e),o})(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 T}}q.finalized=!0,q._$litElement$=!0,null===(X=globalThis.litElementHydrateSupport)||void 0===X||X.call(globalThis,{LitElement:q});const Z=globalThis.litElementPolyfillSupport;null==Z||Z({LitElement:q}),(null!==(Y=globalThis.litElementVersions)&&void 0!==Y?Y:globalThis.litElementVersions=[]).push("3.2.2");class J{static deleteNodesKeyboardKey="Delete";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 linkCurveHeight=15;static linkCurveWidth=80;static linkMinWidth=100;static linkRightSVGPath=(e,t,n)=>{let i=100-e;return`M ${e} 0 C ${t} 0, ${n} 0, 50 50 S ${i-t+e} 100, ${i} 100`};static maxZoom=7;static minZoom=-12;static mouseWheelFactor=.2;static nodeDeleteEventName="ueb-node-delete";static nodeDragEventName="ueb-node-drag";static nodeDragLocalEventName="ueb-node-drag-local";static nodeName=(e,t)=>`${e}_${t}`;static nodeRadius=8;static nodeReflowEventName="ueb-node-reflow";static pinColor={bool:r``,default:r``,exec:r``,name:r``,real:r``,string:r``,"/Script/CoreUObject.LinearColor":r``};static selectAllKeyboardKey="(bCtrl=True,Key=A)";static trackingMouseEventName={begin:"ueb-tracking-mouse-begin",end:"ueb-tracking-mouse-end"};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 Q{#e;get target(){return this.#e}#t;get blueprint(){return this.#t}options;constructor(e,t,n){this.#e=e,this.#t=t,n.consumeEvent??=!1,n.listenOnFocus??=!1,n.unlistenOnTextEdit??=!1,this.options=n;let i=this;this.listenHandler=e=>i.listenEvents(),this.unlistenHandler=e=>i.unlistenEvents(),this.options.listenOnFocus&&(this.blueprint.addEventListener(J.focusEventName.begin,this.listenHandler),this.blueprint.addEventListener(J.focusEventName.end,this.unlistenHandler)),this.options.unlistenOnTextEdit&&(this.blueprint.addEventListener(J.editTextEventName.begin,this.unlistenHandler),this.blueprint.addEventListener(J.editTextEventName.end,this.listenHandler))}unlistenDOMElement(){this.unlistenEvents(),this.blueprint.removeEventListener(J.focusEventName.begin,this.listenHandler),this.blueprint.removeEventListener(J.focusEventName.end,this.unlistenHandler),this.blueprint.removeEventListener(J.editTextEventName.begin,this.unlistenHandler),this.blueprint.removeEventListener(J.editTextEventName.end,this.listenHandler)}listenEvents(){}unlistenEvents(){}}class ee{#n;constructor(e){this.#n=e}calculate(e){return this.#n(e)}}class te{#i;get type(){return this.#i}set type(e){this.#i=e}#r=!0;get showDefault(){return this.#r}set showDefault(e){this.#r=e}#s;get value(){return this.#s}set value(e){this.#s=e}#o;get serialized(){return this.#o}set serialized(e){this.#o=e}static sanitize(e,t){return void 0===t&&(t=e?.constructor),t&&!(e?.constructor===t||e instanceof t)&&(e=new t(e)),(e instanceof Boolean||e instanceof Number||e instanceof String)&&(e=e.valueOf()),e}constructor(e,t=!0,n,i=!1){void 0===n&&(n=e instanceof Array?[]:i?"":te.sanitize(new e)),this.#i=e,this.#r=t,this.#s=n,this.#o=i}}class ne{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 clamp(e,t,n){return Math.min(Math.max(e,t),n)}static getScale(e){return Number(getComputedStyle(e).getPropertyValue("--ueb-scale"))}static minDecimals(e,t=1){const n=e*10**t;return Math.abs(n%1)>Number.EPSILON?e.toString():e.toFixed(t)}static convertLocation(e,t){const n=1/ne.getScale(t),i=t.getBoundingClientRect();return[Math.round((e[0]-i.x)*n),Math.round((e[1]-i.y)*n)]}static isSerialized(e,t,n=ne.objectGet(e.constructor.attributes,t)){return n instanceof ee?ne.isSerialized(e,t,n.calculate(e)):n instanceof te&&(!!n.serialized||ne.isSerialized(e,t,n.type))}static objectGet(e,t,n){if(void 0!==e){if(!(t instanceof Array))throw new TypeError("Expected keys to be an array.");return 0!=t.length&&t[0]in e&&void 0!==e[t[0]]?1==t.length?e[t[0]]:ne.objectGet(e[t[0]],t.slice(1),n):n}}static objectSet(e,t,n,i=!1,r=Object){if(!(t instanceof Array))throw new TypeError("Expected keys to be an array.");if(1==t.length){if(i||t[0]in e||void 0===e[t[0]])return e[t[0]]=n,!0}else if(t.length>0)return!i||e[t[0]]instanceof Object||(e[t[0]]=new r),ne.objectSet(e[t[0]],t.slice(1),n,i,r);return!1}static equals(e,t){return(e=te.sanitize(e))===(t=te.sanitize(t))||(e instanceof Array&&t instanceof Array?e.length==t.length&&!e.find(((e,n)=>!ne.equals(e,t[n]))):void 0)}static firstCapital(e){return e.charAt(0).toUpperCase()+e.substring(1)}static getType(e){if(null===e)return null;let t=e?.constructor;switch(t){case te:return ne.getType(e.type);case Function:return e;default:return t}}static snapToGrid(e,t){return 1===t?e:[t*Math.round(e[0]/t),t*Math.round(e[1]/t)]}static mergeArrays(e=[],t=[]){let n=[];for(let i=0;i{t(this[e])}))}}})}return!0}unsubscribe(e,t){let n=this.#l.get(e);if(!n?.includes(t))return!1;if(n.splice(n.indexOf(t),1),0==n.length){const t=Symbol.for(e+"Storage"),n=Symbol.for(e+"ValInfo"),i=this[n][0];this[n][1],Object.defineProperty(i?Object.getPrototypeOf(this):this,e,Object.getOwnPropertyDescriptor(i?Object.getPrototypeOf(this):this,t)),delete this[n],delete this[t]}return!0}}{static attributes={};constructor(e){super();const t=(e,n,i,r="")=>{for(let s of ne.mergeArrays(Object.getOwnPropertyNames(n),Object.getOwnPropertyNames(i??{}))){let o=ne.objectGet(i,[s]),a=n[s],l=ne.getType(a);if(a instanceof ee&&(a=a.calculate(this),l=ne.getType(a)),s in n?s in i||void 0===a||a instanceof te&&!a.showDefault||console.warn(`${this.constructor.name}.properties will add property ${r}${s} not defined in the serialized data`):console.warn(`Property ${r}${s} in the serialized data is not defined in ${this.constructor.name}.properties`),l!==Object)if(void 0===o){if(a instanceof te){if(!a.showDefault){e[s]=void 0;continue}a.serialized?a="":(l=a.type,a=a.value)}a instanceof Array&&(a=[]),e[s]=te.sanitize(a,l)}else o?.constructor===String&&a instanceof te&&a.serialized&&a.type!==String&&(o=ie.getSerializer(a.type).deserialize(o)),e[s]=te.sanitize(o,ne.getType(a));else e[s]={},t(e[s],n[s],i[s],s+".")}},n=this.constructor.attributes;e.constructor!==Object&&1==Object.getOwnPropertyNames(n).length&&(e={[Object.getOwnPropertyNames(n)[0]]:e}),t(this,n,e)}}class se extends re{static attributes={type:String,path:String}}class oe extends re{static attributes={MemberParent:se,MemberName:""}}class ae extends re{static attributes={value:String};static generateGuid(e=!0){let t=new Uint32Array(4);!0===e&&crypto.getRandomValues(t);let n="";return t.forEach((e=>{n+=("0".repeat(8)+e.toString(16).toUpperCase()).slice(-8)})),new ae({value:n})}valueOf(){return this.value}toString(){return this.value}}class le extends re{static attributes={value:String};static attributeConverter={fromAttribute:(e,t)=>new le(e),toAttribute:(e,t)=>e.toString()};valueOf(){return this.value}toString(){return this.value}}class ue extends re{static attributes={value:Number};constructor(e=0){super(e),this.value=Math.round(this.value)}valueOf(){return this.value}toString(){return this.value.toString()}}class ce extends re{static lookbehind="INVTEXT";static attributes={value:String}}class de extends re{static attributes={ActionName:"",bShift:!1,bCtrl:!1,bAlt:!1,bCmd:!1,Key:le};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)}}class he extends re{static attributes={R:Number,G:Number,B:Number,A:Number};toString(){return ne.printLinearColor(this)}}class pe extends re{static lookbehind="NSLOCTEXT";static attributes={namespace:String,key:String,value:String}}class ge extends re{static attributes={value:String};valueOf(){return this.value}toString(){return this.value}}class me extends re{static attributes={objectName:ge,pinGuid:ae}}class fe extends re{static attributes={X:Number,Y:Number,Z:Number}}class ve extends re{static#u={"/Script/CoreUObject.LinearColor":he,"/Script/CoreUObject.Vector":fe,bool:Boolean,exec:String,name:String,real:Number,string:String};static get typeEntityMap(){return ve.#u}static lookbehind="Pin";static attributes={PinId:ae,PinName:"",PinFriendlyName:new te(pe,!1,null),PinToolTip:"",Direction:new te(String,!1,""),PinType:{PinCategory:"",PinSubCategory:"",PinSubCategoryObject:se,PinSubCategoryMemberReference:null,PinValueType:null,ContainerType:se,bIsReference:!1,bIsConst:!1,bIsWeakPointer:!1,bIsUObjectWrapper:!1,bSerializeAsSinglePrecisionFloat:!1},LinkedTo:new te([me],!1),DefaultValue:new ee((e=>new te(ve.typeEntityMap[e.getType()]??String,!1,void 0,!0))),AutogeneratedDefaultValue:new te(String,!1),DefaultObject:new te(se,!1,null),PersistentGuid:ae,bHidden:!1,bNotConnectable:!1,bDefaultValueIsReadOnly:!1,bDefaultValueIsIgnored:!1,bAdvancedView:!1,bOrphanedPin:!1};getType(){return"struct"==this.PinType.PinCategory?this.PinType.PinSubCategoryObject.path:this.PinType.PinCategory}getDefaultValue(){return this.DefaultValue??""}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){this.LinkedTo;const n=this.LinkedTo?.find((n=>n.objectName==e&&n.pinGuid.valueOf()==t.PinId.valueOf()));return!n&&((this.LinkedTo??(this.LinkedTo=[])).push(new me({objectName:e,pinGuid:t.PinId})),!0)}unlinkFrom(e,t){const n=this.LinkedTo?.findIndex((n=>n.objectName==e&&n.pinGuid.valueOf()==t.PinId.valueOf()));return n>=0&&(1==this.LinkedTo.length?this.LinkedTo=void 0:this.LinkedTo.splice(n,1),!0)}getSubCategory(){return this.PinType.PinSubCategoryObject.path}}class be extends re{static attributes={MemberName:String,MemberGuid:ae,bSelfContext:!1}}class ye extends re{static attributes={Class:se,Name:"",bIsPureFunc:new te(Boolean,!1,!1),VariableReference:new te(be,!1,null),FunctionReference:new te(oe,!1,null),EventReference:new te(oe,!1,null),TargetType:new te(se,!1,null),NodePosX:ue,NodePosY:ue,AdvancedPinDisplay:new te(le,!1,null),EnabledState:new te(le,!1,null),NodeGuid:ae,ErrorType:new te(ue,!1),ErrorMsg:new te(String,!1,""),CustomProperties:[ve]};static nameRegex=/(\w+)_(\d+)/;getObjectName(e=!1){return e?this.getNameAndCounter()[0]:this.Name}getNameAndCounter(){const e=this.getObjectName(!1).match(ye.nameRegex);return e&&3==e.length?[e[1],parseInt(e[2])]:["",0]}getDisplayName(){let e=this.FunctionReference?.MemberName;return e?(e=ne.formatStringName(e),e):(e=ne.formatStringName(this.getNameAndCounter()[0]),e)}getCounter(){return this.getNameAndCounter()[1]}}"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self&&self;function Ee(e){return e&&e.__esModule&&Object.prototype.hasOwnProperty.call(e,"default")?e.default:e}var we={exports:{}};"undefined"!=typeof self&&self;var Pe=Ee(we.exports=function(e){var t={};function n(i){if(t[i])return t[i].exports;var r=t[i]={i:i,l:!1,exports:{}};return e[i].call(r.exports,r,r.exports,n),r.l=!0,r.exports}return n.m=e,n.c=t,n.d=function(e,t,i){n.o(e,t)||Object.defineProperty(e,t,{configurable:!1,enumerable:!0,get:i})},n.r=function(e){Object.defineProperty(e,"__esModule",{value:!0})},n.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return n.d(t,"a",t),t},n.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},n.p="",n(n.s=0)}([function(e,t,n){function i(e){if(!(this instanceof i))return new i(e);this._=e}var r=i.prototype;function s(e,t){for(var n=0;n>7),buf:function(e){var t=o((function(e,t,n,i){return e.concat(n===i.length-1?Buffer.from([t,0]).readUInt16BE(0):i.readUInt16BE(n))}),[],e);return Buffer.from(a((function(e){return(e<<1&65535)>>8}),t))}(n.buf)}})),n}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 d(e){c();var t=o((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 n,r=t/8,s=(n=function(e){return e>48},o((function(e,t){return e||(n(t)?t:e)}),null,e));if(s)throw new Error(s+" bit range requested exceeds 48 bit (6 byte) Number max.");return new i((function(t,n){var i=r+n;return i>t.length?P(n,r.toString()+" bytes"):w(i,o((function(e,t){var n=l(t,e.buf);return{coll:e.coll.concat(n.v),buf:n.buf}}),{coll:[],buf:t.slice(n,i)},e).coll)}))}function h(e,t){return new i((function(n,i){return c(),i+t>n.length?P(i,t+" bytes for "+e):w(i+t,n.slice(i,i+t))}))}function p(e,t){if("number"!=typeof(n=t)||Math.floor(n)!==n||t<0||t>6)throw new Error(e+" requires integer length in range [0, 6].");var n}function g(e){return p("uintBE",e),h("uintBE("+e+")",e).map((function(t){return t.readUIntBE(0,e)}))}function m(e){return p("uintLE",e),h("uintLE("+e+")",e).map((function(t){return t.readUIntLE(0,e)}))}function f(e){return p("intBE",e),h("intBE("+e+")",e).map((function(t){return t.readIntBE(0,e)}))}function v(e){return p("intLE",e),h("intLE("+e+")",e).map((function(t){return t.readIntLE(0,e)}))}function b(e){return e instanceof i}function y(e){return"[object Array]"==={}.toString.call(e)}function E(e){return u()&&Buffer.isBuffer(e)}function w(e,t){return{status:!0,index:e,value:t,furthest:-1,expected:[]}}function P(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 n=e.furthest===t.furthest?function(e,t){if(function(){if(void 0!==i._supportsSet)return i._supportsSet;var e="undefined"!=typeof Set;return i._supportsSet=e,e}()&&Array.from){for(var n=new Set(e),r=0;r=0;){if(o in n){i=n[o].line,0===s&&(s=n[o].lineStart);break}("\n"===e.charAt(o)||"\r"===e.charAt(o)&&"\n"!==e.charAt(o+1))&&(r++,0===s&&(s=o+1)),o--}var a=i+r,l=t-s;return n[t]={line:a,lineStart:s},{offset:t,line:a+1,column:l+1}}function C(e){if(!b(e))throw new Error("not a parser: "+e)}function N(e,t){return"string"==typeof e?e.charAt(t):e[t]}function A(e){if("number"!=typeof e)throw new Error("not a number: "+e)}function $(e){if("function"!=typeof e)throw new Error("not a function: "+e)}function L(e){if("string"!=typeof e)throw new Error("not a string: "+e)}var O=2,T=3,D=8,M=5*D,_=4*D,H=" ";function I(e,t){return new Array(t+1).join(e)}function j(e,t,n){var i=t-e.length;return i<=0?e:I(n,i)+e}function z(e,t,n,i){return{from:e-t>0?e-t:0,to:e+n>i?i:e+n}}function B(e,t){var n,i,r,s,l,u=t.index,c=u.offset,d=1;if(c===e.length)return"Got the end of the input";if(E(e)){var h=c-c%D,p=c-h,g=z(h,M,_+D,e.length),m=a((function(e){return a((function(e){return j(e.toString(16),2,"0")}),e)}),function(e,t){var n=e.length,i=[],r=0;if(n<=t)return[e.slice()];for(var s=0;s=4&&(n+=1),d=2,r=a((function(e){return e.length<=4?e.join(" "):e.slice(0,4).join(" ")+" "+e.slice(4).join(" ")}),m),(l=(8*(s.to>0?s.to-1:s.to)).toString(16).length)<2&&(l=2)}else{var f=e.split(/\r\n|[\n\r\u2028\u2029]/);n=u.column-1,i=u.line-1,s=z(i,O,T,f.length),r=f.slice(s.from,s.to),l=s.to.toString().length}var v=i-s.from;return E(e)&&(l=(8*(s.to>0?s.to-1:s.to)).toString(16).length)<2&&(l=2),o((function(t,i,r){var o,a=r===v,u=a?"> ":H;return o=E(e)?j((8*(s.from+r)).toString(16),l,"0"):j((s.from+r+1).toString(),l," "),[].concat(t,[u+o+" | "+i],a?[H+I(" ",l)+" | "+j("",n," ")+I("^",d)]:[])}),[],r).join("\n")}function F(e,t){return["\n","-- PARSING FAILED "+I("-",50),"\n\n",B(e,t),"\n\n",(n=t.expected,1===n.length?"Expected:\n\n"+n[0]:"Expected one of the following: \n\n"+n.join(", ")),"\n"].join("");var n}function U(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,n=0;n=2?A(t):t=0;var n=function(e){return RegExp("^(?:"+e.source+")",U(e))}(e),r=""+e;return i((function(e,i){var s=n.exec(e.slice(i));if(s){if(0<=t&&t<=s.length){var o=s[0],a=s[t];return w(i+o.length,a)}return P(i,"valid match group (0 to "+s.length+") in "+r)}return P(i,r)}))}function q(e){return i((function(t,n){return w(n,e)}))}function Z(e){return i((function(t,n){return P(n,e)}))}function J(e){if(b(e))return i((function(t,n){var i=e._(t,n);return i.index=n,i.value="",i}));if("string"==typeof e)return J(X(e));if(e instanceof RegExp)return J(Y(e));throw new Error("not a string, regexp, or parser: "+e)}function Q(e){return C(e),i((function(t,n){var i=e._(t,n),r=t.slice(n,i.index);return i.status?P(n,'not "'+r+'"'):w(n,null)}))}function ee(e){return $(e),i((function(t,n){var i=N(t,n);return n=e.length?P(t,"any character/byte"):w(t+1,N(e,t))})),se=i((function(e,t){return w(e.length,e.slice(t))})),oe=i((function(e,t){return t=0})).desc(t)},i.optWhitespace=de,i.Parser=i,i.range=function(e,t){return ee((function(n){return e<=n&&n<=t})).desc(e+"-"+t)},i.regex=Y,i.regexp=Y,i.sepBy=K,i.sepBy1=W,i.seq=G,i.seqMap=R,i.seqObj=function(){for(var e,t={},n=0,r=(e=arguments,Array.prototype.slice.call(e)),s=r.length,o=0;o255)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 i((function(n,i){var r=N(n,i);return r===e?w(i+1,r):P(i,t)}))},buffer:function(e){return h("buffer",e).map((function(e){return Buffer.from(e)}))},encodedString:function(e,t){return h("string",t).map((function(t){return t.toString(e)}))},uintBE:g,uint8BE:g(1),uint16BE:g(2),uint32BE:g(4),uintLE:m,uint8LE:m(1),uint16LE:m(2),uint32LE:m(4),intBE:f,int8BE:f(1),int16BE:f(2),int32BE:f(4),intLE:v,int8LE:v(1),int16LE:v(2),int32LE:v(4),floatBE:h("floatBE",4).map((function(e){return e.readFloatBE(0)})),floatLE:h("floatLE",4).map((function(e){return e.readFloatLE(0)})),doubleBE:h("doubleBE",8).map((function(e){return e.readDoubleBE(0)})),doubleLE:h("doubleLE",8).map((function(e){return e.readDoubleLE(0)}))},e.exports=i}]));let Se=Pe;class ke{static getGrammarForType(e,t,n){if(t instanceof te){let i=ke.getGrammarForType(e,t.type,n);return!t.serialized||t.type instanceof String||(i=i.wrap(Se.string('"'),Se.string('"'))),i}switch(ne.getType(t)){case Boolean:return e.Boolean;case Number:return e.Number;case ue:return e.Integer;case String:return e.String;case ae:return e.Guid;case le:return e.Identifier;case se:return e.Reference;case pe:return e.LocalizedText;case ce:return e.InvariantText;case me:return e.PinReference;case fe:return e.Vector;case he:return e.LinearColor;case oe:return e.FunctionReference;case ve:return e.Pin;case Array:return Se.seqMap(Se.string("("),t.map((t=>ke.getGrammarForType(e,ne.getType(t)))).reduce(((t,n)=>n&&t!==e.AttributeAnyValue?t.or(n):e.AttributeAnyValue)).trim(Se.optWhitespace).sepBy(Se.string(",")).skip(Se.regex(/,?\s*/)),Se.string(")"),((e,t,n)=>t));default:return n}}static createPropertyGrammar=(e,t,n=Se.string("=").trim(Se.optWhitespace))=>e.AttributeName.skip(n).chain((n=>{const i=n.split("."),r=ne.objectGet(t.attributes,i);return ke.getGrammarForType(e,r,e.AttributeAnyValue).map((e=>t=>ne.objectSet(t,i,e,!0)))}));static createEntityGrammar=(e,t)=>Se.seqMap(t.lookbehind?Se.seq(Se.string(t.lookbehind),Se.optWhitespace,Se.string("(")):Se.string("("),ke.createPropertyGrammar(e,t).trim(Se.optWhitespace).sepBy(Se.string(",")).skip(Se.regex(/,?/).then(Se.optWhitespace)),Se.string(")"),((e,n,i)=>{let r={};return n.forEach((e=>e(r))),new t(r)}));InlineWhitespace=e=>Se.regex(/[^\S\n]+/).desc("inline whitespace");InlineOptWhitespace=e=>Se.regex(/[^\S\n]*/).desc("inline optional whitespace");MultilineWhitespace=e=>Se.regex(/[^\S\n]*\n\s*/).desc("whitespace with at least a newline");Null=e=>Se.seq(Se.string("("),e.InlineOptWhitespace,Se.string(")")).map((e=>null)).desc("null: ()");Boolean=e=>Se.alt(Se.string("True"),Se.string("true"),Se.string("False"),Se.string("false")).map((e=>"true"===e.toLocaleLowerCase())).desc("either True or False");HexDigit=e=>Se.regex(/[0-9a-fA-f]/).desc("hexadecimal digit");Number=e=>Se.regex(/[\-\+]?[0-9]+(?:\.[0-9]+)?/).map(Number).desc("a number");NaturalNumber=e=>Se.regex(/0|[1-9]\d*/).map(Number).desc("a natural number");ColorNumber=e=>e.NaturalNumber.assert((e=>0<=e&&e<256),"the color must be between 0 and 256 excluded");Word=e=>Se.regex(/[a-zA-Z]+/).desc("a word");String=e=>Se.regex(/(?:[^"\\]|\\.)*/).wrap(Se.string('"'),Se.string('"')).map(ne.decodeString).desc('string (with possibility to escape the quote using ")');ReferencePath=e=>Se.seq(Se.string("/"),e.PathSymbol.map((e=>e.toString())).sepBy1(Se.string(".")).tieWith(".")).tie().atLeast(2).tie().desc('a path (words with possibly underscore, separated by ".", separated by "/")');AttributeName=e=>e.Word.sepBy1(Se.string(".")).tieWith(".").desc('words separated by ""');None=e=>Se.string("None").map((e=>new se({type:"None",path:""}))).desc("none");Integer=e=>Se.regex(/[\-\+]?[0-9]+/).map((e=>new ue(e))).desc("an integer");Guid=e=>e.HexDigit.times(32).tie().map((e=>new ae({value:e}))).desc("32 digit hexadecimal value");Identifier=e=>Se.regex(/\w+/).map((e=>new le(e)));PathSymbol=e=>Se.regex(/[0-9\w]+/).map((e=>new ge({value:e})));Reference=e=>Se.alt(e.None,...[e.ReferencePath.map((e=>new se({type:"",path:e})))].flatMap((e=>[e,e.trim(Se.string('"'))])),Se.seqMap(e.Word,Se.optWhitespace,Se.alt(...[e.ReferencePath].flatMap((e=>[e.wrap(Se.string('"'),Se.string('"')),e.wrap(Se.string("'\""),Se.string("\"'"))]))),((e,t,n)=>new se({type:e,path:n}))),e.Word.map((e=>new se({type:e,path:""}))));LocalizedText=e=>Se.seqMap(Se.string(pe.lookbehind).skip(Se.optWhitespace).skip(Se.string("(")),e.String.trim(Se.optWhitespace),Se.string(","),e.String.trim(Se.optWhitespace),Se.string(","),e.String.trim(Se.optWhitespace),Se.string(")"),((e,t,n,i,r,s,o)=>new pe({namespace:t,key:i,value:s})));InvariantText=e=>e.String.trim(Se.optWhitespace).wrap(Se.string(ce.lookbehind).skip(Se.optWhitespace).skip(Se.string("(")),Se.string(")")).map((e=>new ce({value:e})));AttributeAnyValue=e=>Se.alt(e.Null,e.None,e.Boolean,e.Number,e.Integer,e.String,e.Guid,e.LocalizedText,e.InvariantText,e.Reference,e.Vector,e.LinearColor);PinReference=e=>Se.seqMap(e.PathSymbol,Se.whitespace,e.Guid,((e,t,n)=>new me({objectName:e,pinGuid:n})));Vector=e=>ke.createEntityGrammar(e,fe);LinearColor=e=>ke.createEntityGrammar(e,he);FunctionReference=e=>ke.createEntityGrammar(e,oe);KeyBinding=e=>Se.alt(e.Identifier.map((e=>new de({Key:e}))),ke.createEntityGrammar(e,de));Pin=e=>ke.createEntityGrammar(e,ve);CustomProperties=e=>Se.string("CustomProperties").then(Se.whitespace).then(e.Pin).map((e=>t=>{let n=ne.objectGet(t,["CustomProperties"],[]);n.push(e),ne.objectSet(t,["CustomProperties"],n,!0)}));Object=e=>Se.seqMap(Se.seq(Se.string("Begin"),Se.whitespace,Se.string("Object"),Se.whitespace),Se.alt(e.CustomProperties,ke.createPropertyGrammar(e,ye)).sepBy1(Se.whitespace),Se.seq(e.MultilineWhitespace,Se.string("End"),Se.whitespace,Se.string("Object")),((e,t,n)=>{let i={};return t.forEach((e=>e(i))),new ye(i)}));MultipleObject=e=>e.Object.sepBy1(Se.whitespace).trim(Se.optWhitespace);LinearColorFromHex=e=>Se.string("#").then(e.HexDigit.times(2).tie().times(3,4)).trim(Se.optWhitespace).map((([e,t,n,i])=>new he({R:parseInt(e,16)/255,G:parseInt(t,16)/255,B:parseInt(n,16)/255,A:i?parseInt(i,16)/255:1})));LinearColorFromRGBList=e=>Se.seqMap(e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber.map(Number),((e,t,n,i,r)=>new he({R:e/255,G:n/255,B:r/255,A:1})));LinearColorFromRGB=e=>Se.string("rgb").then(e.LinearColorFromRGBList.wrap(Se.regex(/\(\s*/),Se.regex(/\s*\)/)));LinearColorFromRGBA=e=>Se.string("rgba").then(Se.seqMap(e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber,Se.string(",").skip(Se.optWhitespace),e.ColorNumber.map(Number),Se.string(",").skip(Se.optWhitespace),Se.regex(/0?\.\d+|[01]/).map(Number),((e,t,n,i,r,s,o)=>new he({R:e/255,G:n/255,B:r/255,A:o}))).wrap(Se.regex(/\(\s*/),Se.regex(/\s*\)/)));LinearColorFromAnyColor=e=>Se.alt(e.LinearColorFromRGBList,e.LinearColorFromHex,e.LinearColorFromRGB,e.LinearColorFromRGBA)}class xe{static grammar=Pe.createLanguage(new ke);constructor(e,t,n,i,r,s){this.entityType=e,this.prefix=t??"",this.separator=n??",",this.trailingSeparator=i??!1,this.attributeValueConjunctionSign=r??"=",this.attributeKeyPrinter=s??(e=>e.join("."))}deserialize(e){return this.read(e)}serialize(e,t,n=e){return this.write(n,e,t)}read(e){throw new Error("Not implemented")}write(e,t,n){throw new Error("Not implemented")}writeValue(e,t,n,i){const r=ie.getSerializer(ne.getType(t));if(!r)throw new Error("Unknown value type, a serializer must be registered in the SerializerFactory class");return r.write(e,t,i)}subWrite(e,t,n,i){let r="",s=t.concat("");const o=s.length-1;for(const t of Object.getOwnPropertyNames(n)){s[o]=t;const a=n[t];if(a?.constructor===Object)r+=(r.length?this.separator:"")+this.subWrite(e,s,a,i);else if(void 0!==a&&this.showProperty(e,n,s,a)){const t=ne.isSerialized(e,s);r+=(r.length?this.separator:"")+this.prefix+this.attributeKeyPrinter(s)+this.attributeValueConjunctionSign+(t?`"${this.writeValue(e,a,s,!0)}"`:this.writeValue(e,a,s,i))}}return this.trailingSeparator&&r.length&&1===s.length&&(r+=this.separator),r}showProperty(e,t,n,i){const r=this.entityType.attributes,s=ne.objectGet(r,n);return!(s instanceof te)||(!ne.equals(s.value,i)||s.showDefault)}}class Ce extends xe{constructor(){super(ye," ","\n",!1)}showProperty(e,t,n,i){switch(n.toString()){case"Class":case"Name":case"CustomProperties":return!1}return super.showProperty(e,t,n,i)}read(e){const t=xe.grammar.Object.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}readMultiple(e){const t=xe.grammar.MultipleObject.parse(e);if(!t.status)throw new Error("Error when trying to parse the object.");return t.value}write(e,t,n){return`Begin Object Class=${t.Class.path} Name=${this.writeValue(e,t.Name,["Name"],n)}\n${this.subWrite(e,[],t,n)+t.CustomProperties.map((e=>this.separator+this.prefix+"CustomProperties "+ie.getSerializer(ve).serialize(e))).join("")}\nEnd Object\n`}}class Ne extends Q{#c;constructor(e,t,n={}){n.listenOnFocus=!0,n.unlistenOnTextEdit=!0,super(e,t,n),this.serializer=new Ce;let i=this;this.#c=e=>i.copied()}listenEvents(){document.body.addEventListener("copy",this.#c)}unlistenEvents(){document.body.removeEventListener("copy",this.#c)}copied(){const e=this.blueprint.getNodes(!0).map((e=>this.serializer.serialize(e.entity,!1))).join("\n\n");navigator.clipboard.writeText(e)}}class Ae{static styles=r``;#d=[];get inputObjects(){return this.#d}constructed(e){}connectedCallback(e){}willUpdate(e,t){}update(e,t){}render(e){return O``}firstUpdated(e,t){}updated(e,t){}inputSetup(e){this.#d=this.createInputObjects(e)}cleanup(e){this.#d.forEach((e=>e.unlistenDOMElement()))}createInputObjects(e){return[]}}class $e extends Q{#h;constructor(e,t,n={}){n.activateAnyKey??=!1,n.activationKeys??=[],n.listenOnFocus??=!0,n.unlistenOnTextEdit??=!0,n.activationKeys instanceof Array||(n.activationKeys=[n.activationKeys]),n.activationKeys=n.activationKeys.map((e=>{if(e instanceof de)return e;if(e.constructor===String){const t=xe.grammar.KeyBinding.parse(e);if(t.status)return t.value}throw new Error("Unexpected key value")})),super(e,t,n),this.#h=this.options.activationKeys??[];let i=this;this.keyDownHandler=e=>{(this.options.activateAnyKey||i.#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&&J.Keys[t.Key]==e.code)))&&(n.consumeEvent&&e.stopImmediatePropagation(),i.fire(),document.removeEventListener("keydown",i.keyDownHandler),document.addEventListener("keyup",i.keyUpHandler))},this.keyUpHandler=e=>{(this.options.activateAnyKey||i.#h.some((t=>t.bShift&&"Shift"==e.key||t.bCtrl&&"Control"==e.key||t.bAlt&&"Alt"==e.key||t.bCmd&&"Meta"==e.key||J.Keys[t.Key]==e.code)))&&(n.consumeEvent&&e.stopImmediatePropagation(),i.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 Le extends $e{constructor(e,t,n={}){n.activationKeys=J.deleteNodesKeyboardKey,super(e,t,n)}fire(){this.blueprint.removeGraphElement(...this.blueprint.getNodes(!0))}}class Oe extends Q{constructor(e,t,n){super(e,t,n),this.movementSpace=this.blueprint?.getGridDOMElement()??document.documentElement}locationFromEvent(e){const t=ne.convertLocation([e.clientX,e.clientY],this.movementSpace);return this.blueprint.compensateTranslation(t)}}class Te extends Oe{#p;#g;constructor(e,t,n){n.listenOnFocus=!0,super(e,t,n),this.looseTarget=n?.looseTarget??!0;let i=this;this.#p=e=>{e.preventDefault();const t=i.locationFromEvent(e);i.wheel(Math.sign(e.deltaY*J.mouseWheelFactor),t)},this.#g=e=>e.preventDefault(),this.blueprint.focused&&this.movementSpace.addEventListener("wheel",this.#p,!1)}listenEvents(){this.movementSpace.addEventListener("wheel",this.#p,!1),this.movementSpace.parentElement?.addEventListener("wheel",this.#g)}unlistenEvents(){this.movementSpace.removeEventListener("wheel",this.#p,!1),this.movementSpace.parentElement?.removeEventListener("wheel",this.#g)}wheel(e,t){}}class De extends Te{#m=!1;get enableZoonIn(){return this.#m}set enableZoonIn(e){(e=Boolean(e))!=this.#m&&(this.#m=e)}wheel(e,t){let n=this.blueprint.getZoom();e=-e,!this.enableZoonIn&&0==n&&e>0||(n+=e,this.blueprint.setZoom(n,t))}}class Me extends $e{#f;constructor(e,t,n={}){n.activationKeys=J.enableZoomIn,super(e,t,n)}fire(){this.#f=this.blueprint.getInputObject(De),this.#f.enableZoonIn=!0}unfire(){this.#f.enableZoonIn=!1}}class _e extends $e{constructor(e,t,n={}){n.activationKeys=J.selectAllKeyboardKey,super(e,t,n)}fire(){this.blueprint.selectAll()}}class He extends Oe{#v;#b;#y;#E;#w=!1;started=!1;constructor(e,t,n={}){n.clickButton??=0,n.consumeEvent??=!0,n.exitAnyButton??=!0,n.looseTarget??=!1,n.moveEverywhere??=!1,super(e,t,n),this.clickedPosition=[0,0];const i=this.options.moveEverywhere?document.documentElement:this.movementSpace;let r=this;this.#v=e=>{if(r.blueprint.setFocused(!0),e.button===r.options.clickButton)(r.options.looseTarget||e.target==e.currentTarget)&&(r.options.consumeEvent&&e.stopImmediatePropagation(),i.addEventListener("mousemove",r.#b),document.addEventListener("mouseup",r.#E),r.clickedPosition=r.locationFromEvent(e),r.clicked(r.clickedPosition));else r.options.exitAnyButton||r.#E(e)},this.#b=e=>{r.options.consumeEvent&&e.stopImmediatePropagation(),i.removeEventListener("mousemove",r.#b),i.addEventListener("mousemove",r.#y);const t=r.getEvent(J.trackingMouseEventName.begin);r.#w=0==r.target.dispatchEvent(t);const n=r.locationFromEvent(e);r.startDrag(n),r.started=!0},this.#y=e=>{r.options.consumeEvent&&e.stopImmediatePropagation();const t=r.locationFromEvent(e),n=[e.movementX,e.movementY];r.dragTo(t,n),r.#w&&(r.blueprint.mousePosition=r.locationFromEvent(e))},this.#E=e=>{if(!r.options.exitAnyButton||e.button==r.options.clickButton){if(r.options.consumeEvent&&e.stopImmediatePropagation(),i.removeEventListener("mousemove",r.#b),i.removeEventListener("mousemove",r.#y),document.removeEventListener("mouseup",r.#E),r.started&&r.endDrag(),r.unclicked(),r.#w){const e=r.getEvent(J.trackingMouseEventName.end);r.target.dispatchEvent(e),r.#w=!1}r.started=!1}},this.listenEvents()}listenEvents(){this.target.addEventListener("mousedown",this.#v),2==this.options.clickButton&&this.target.addEventListener("contextmenu",(e=>e.preventDefault()))}unlistenEvents(){this.target.removeEventListener("mousedown",this.#v)}getEvent(e){return new CustomEvent(e,{detail:{tracker:this},bubbles:!0,cancelable:!0})}clicked(e){}startDrag(e){}dragTo(e,t){}endDrag(){}unclicked(e){}}class Ie extends He{startDrag(){this.blueprint.scrolling=!0}dragTo(e,t){this.blueprint.scrollDelta([-t[0],-t[1]])}endDrag(){this.blueprint.scrolling=!1}}class je extends Oe{#P=null;#S;#k;#x;constructor(e,t,n={}){n.listenOnFocus=!0,super(e,t,n);let i=this;this.#S=e=>{e.preventDefault(),i.blueprint.mousePosition=i.locationFromEvent(e)},this.#k=e=>{i.#P||(e.preventDefault(),this.#P=e.detail.tracker,i.unlistenMouseMove())},this.#x=e=>{i.#P==e.detail.tracker&&(e.preventDefault(),i.#P=null,i.listenMouseMove())}}listenMouseMove(){this.target.addEventListener("mousemove",this.#S)}unlistenMouseMove(){this.target.removeEventListener("mousemove",this.#S)}listenEvents(){this.listenMouseMove(),this.blueprint.addEventListener(J.trackingMouseEventName.begin,this.#k),this.blueprint.addEventListener(J.trackingMouseEventName.end,this.#x)}unlistenEvents(){this.unlistenMouseMove(),this.blueprint.removeEventListener(J.trackingMouseEventName.begin,this.#k),this.blueprint.removeEventListener(J.trackingMouseEventName.end,this.#x)}}class ze extends q{static properties={};#C=[];#t;get blueprint(){return this.#t}set blueprint(e){return this.#t=e}#N;get entity(){return this.#N}set entity(e){this.#N=e}#A;get template(){return this.#A}inputObjects=[];constructor(e,t){super(),this.#N=e,this.#A=t,this.inputObjects=[],this.#A.constructed(this)}createRenderRoot(){return this}connectedCallback(){super.connectedCallback(),this.blueprint=this.closest("ueb-blueprint"),this.template.connectedCallback(this)}willUpdate(e){super.willUpdate(e),this.template.willUpdate(this,e)}update(e){super.update(e),this.template.update(this,e)}render(){return this.template.render(this)}firstUpdated(e){super.firstUpdated(e),this.template.firstUpdated(this,e),this.template.inputSetup(this)}updated(e){super.updated(e),this.template.updated(this,e),this.#C.forEach((t=>t(e))),this.#C=[]}disconnectedCallback(){super.disconnectedCallback(),this.template.cleanup(this)}addNextUpdatedCallbacks(e,t=!1){this.#C.push(e),t&&this.requestUpdate()}isSameGraph(e){return this.blueprint&&this.blueprint==e?.blueprint}getInputObject(e){return this.template.inputObjects.find((t=>t.constructor==e))}}class Be extends ze{static properties={...super.properties,selected:{type:Boolean,attribute:"data-selected",reflect:!0,converter:ne.booleanConverter},locationX:{type:Number,attribute:!1},locationY:{type:Number,attribute:!1}};constructor(...e){super(...e),this.selected=!1,this.locationX=0,this.locationY=0,this.listeningDrag=!1;let t=this;this.dragHandler=e=>t.addLocation(e.detail.value)}connectedCallback(){super.connectedCallback(),this.setSelected(this.selected)}disconnectedCallback(){super.disconnectedCallback(),this.blueprint.removeEventListener(J.nodeDragEventName,this.dragHandler)}setLocation([e,t]){const n=[e-this.locationX,t-this.locationY];if(this.locationX=e,this.locationY=t,this.blueprint){const e=new CustomEvent(J.nodeDragLocalEventName,{detail:{value:n},bubbles:!1,cancelable:!0});this.dispatchEvent(e)}}addLocation([e,t]){this.setLocation([this.locationX+e,this.locationY+t])}setSelected(e=!0){this.selected=e,this.blueprint&&(this.selected?(this.listeningDrag=!0,this.blueprint.addEventListener(J.nodeDragEventName,this.dragHandler)):(this.blueprint.removeEventListener(J.nodeDragEventName,this.dragHandler),this.listeningDrag=!1))}dispatchDragEvent(e){const t=new CustomEvent(J.nodeDragEventName,{detail:{value:e},bubbles:!0,cancelable:!0});this.dispatchEvent(t)}snapToGrid(){const e=ne.snapToGrid([this.locationX,this.locationY],J.gridSize);this.locationX==e[0]&&this.locationY==e[1]||this.setLocation(e)}}class Fe extends He{constructor(e,t,n={}){n.consumeEvent=!0,super(e,t,n)}}class Ue extends ze{static properties={...super.properties,initialPositionX:{type:Number,attribute:!1},initialPositionY:{type:Number,attribute:!1},finaPositionX:{type:Number,attribute:!1},finaPositionY:{type:Number,attribute:!1}};constructor(...e){super(...e),this.initialPositionX=0,this.initialPositionY=0,this.finaPositionX=0,this.finaPositionY=0}setBothLocations([e,t]){this.initialPositionX=e,this.initialPositionY=t,this.finaPositionX=e,this.finaPositionY=t}addSourceLocation([e,t]){this.initialPositionX+=e,this.initialPositionY+=t}addDestinationLocation([e,t]){this.finaPositionX+=e,this.finaPositionY+=t}}class Ge extends Ae{update(e,t){super.update(e,t),t.has("initialPositionX")&&e.style.setProperty("--ueb-from-x",`${e.initialPositionX}`),t.has("initialPositionY")&&e.style.setProperty("--ueb-from-y",`${e.initialPositionY}`),t.has("finaPositionX")&&e.style.setProperty("--ueb-to-x",`${e.finaPositionX}`),t.has("finaPositionY")&&e.style.setProperty("--ueb-to-y",`${e.finaPositionY}`)}}class Re extends Ge{static decreasingValue(e,t){const n=-e*t[0]**2,i=t[1]-n/t[0];return e=>n/e+i}static clampedLine(e,t){if(e[0]>t[0]){const n=e;e=t,t=n}const n=(t[1]-e[1])/(t[0]-e[0]),i=e[1]-n*e[0];return r=>rt[0]?t[1]:n*r+i}static c1DecreasingValue=Re.decreasingValue(-.15,[100,15]);static c2DecreasingValue=Re.decreasingValue(-.06,[500,130]);static c2Clamped=Re.clampedLine([0,100],[200,30]);willUpdate(e,t){super.willUpdate(e,t);const n=Math.max(Math.abs(e.initialPositionX-e.finaPositionX),1),i=Math.max(n,J.linkMinWidth),r=n/i,s=e.originatesFromInput?e.initialPositionX ${""!=e.linkMessageIcon||""!=e.linkMessageText?O``:O``}`}}class Ve extends Ue{static properties={...super.properties,source:{type:String,reflect:!0},destination:{type:String,reflect:!0},dragging:{type:Boolean,attribute:"data-dragging",converter:ne.booleanConverter,reflect:!0},originatesFromInput:{type:Boolean,attribute:!1},svgPathD:{type:String,attribute:!1},linkMessageIcon:{type:String,attribute:!1},linkMessageText:{type:String,attribute:!1}};#$;get sourcePin(){return this.#$}set sourcePin(e){this.#L(e,!1)}#O;get destinationPin(){return this.#O}set destinationPin(e){this.#L(e,!0)}#T;#D;#M;#_;#H;pathElement;constructor(e,t){super({},new Re);const n=this;this.#T=()=>n.remove(),this.#D=e=>n.addSourceLocation(e.detail.value),this.#M=e=>n.addDestinationLocation(e.detail.value),this.#_=e=>n.setSourceLocation(),this.#H=e=>n.setDestinationLocation(),this.source=null,this.destination=null,this.dragging=!1,this.originatesFromInput=!1,this.startPercentage=0,this.svgPathD="",this.startPixels=0,this.linkMessageIcon="",this.linkMessageText="",e&&(this.sourcePin=e,t||(this.finaPositionX=this.initialPositionX,this.finaPositionY=this.initialPositionY)),t&&(this.destinationPin=t,e||(this.initialPositionX=this.finaPositionX,this.initialPositionY=this.finaPositionY)),this.#I()}#L(e,t){const n=()=>t?this.destinationPin:this.sourcePin;if(n()!=e){if(n()){const e=n().getNodeElement();e.removeEventListener(J.nodeDeleteEventName,this.#T),e.removeEventListener(J.nodeDragLocalEventName,t?this.#M:this.#D),e.removeEventListener(J.nodeReflowEventName,t?this.#H:this.#_),this.#j()}if(t?this.#O=e:this.#$=e,n()){const e=n().getNodeElement();e.addEventListener(J.nodeDeleteEventName,this.#T),e.addEventListener(J.nodeDragLocalEventName,t?this.#M:this.#D),e.addEventListener(J.nodeReflowEventName,t?this.#H:this.#_),t?this.setDestinationLocation():(this.setSourceLocation(),this.originatesFromInput=this.sourcePin.isInput()),this.#I()}}}#I(){this.sourcePin&&this.destinationPin&&(this.sourcePin.linkTo(this.destinationPin),this.destinationPin.linkTo(this.sourcePin))}#j(){this.sourcePin&&this.destinationPin&&(this.sourcePin.unlinkFrom(this.destinationPin),this.destinationPin.unlinkFrom(this.sourcePin))}disconnectedCallback(){super.disconnectedCallback(),this.#j(),this.sourcePin=null,this.destinationPin=null}setSourceLocation(e=null){if(null==e){const t=this;if(!this.hasUpdated||!this.sourcePin.hasUpdated)return void Promise.all([this.updateComplete,this.sourcePin.updateComplete]).then((()=>t.setSourceLocation()));e=this.sourcePin.template.getLinkLocation(this.sourcePin)}const[t,n]=e;this.initialPositionX=t,this.initialPositionY=n}setDestinationLocation(e=null){if(null==e){const t=this;if(!this.hasUpdated||!this.destinationPin.hasUpdated)return void Promise.all([this.updateComplete,this.destinationPin.updateComplete]).then((()=>t.setDestinationLocation()));e=this.destinationPin.template.getLinkLocation(this.destinationPin)}this.finaPositionX=e[0],this.finaPositionY=e[1]}startDragging(){this.dragging=!0}finishDragging(){this.dragging=!1}removeMessage(){this.linkMessageIcon="",this.linkMessageText=""}setMessageConvertType(){this.linkMessageIcon="ueb-icon-conver-type",this.linkMessageText=`Convert ${this.sourcePin.pinType} to ${this.destinationPin.pinType}.`}setMessageCorrect(){this.linkMessageIcon="ueb-icon-correct",this.linkMessageText=""}setMessageDirectionsIncompatible(){this.linkMessageIcon="ueb-icon-directions-incompatible",this.linkMessageText="Directions are not compatbile."}setMessagePlaceNode(){this.linkMessageIcon="ueb-icon-place-node",this.linkMessageText="Place a new node."}setMessageReplaceLink(){this.linkMessageIcon="ueb-icon-replace-link",this.linkMessageText="Replace existing input connections."}setMessageSameNode(){this.linkMessageIcon="ueb-icon-same-node",this.linkMessageText="Both are on the same node."}setMEssagetypesIncompatible(){this.linkMessageIcon="ueb-icon-types-incompatible",this.linkMessageText=`${this.sourcePin.pinType} is not compatible with ${this.destinationPin.pinType}.`}}customElements.define("ueb-link",Ve);class Ke extends He{#z;#B;#F;link;enteredPin;linkValid=!1;constructor(e,t,n){super(e,t,n);let i=this;this.#B=e=>{if(!i.enteredPin){i.linkValid=!1,i.enteredPin=e.target;const t=i.enteredPin,n=i.target;t.getNodeElement()==n.getNodeElement()?i.link.setMessageSameNode():t.isOutput()==n.isOutput()||t.isOutput()==n.isOutput()?i.link.setMessageDirectionsIncompatible():i.blueprint.getLinks([t,n]).length?(i.link.setMessageReplaceLink(),i.linkValid=!0):(i.link.setMessageCorrect(),i.linkValid=!0)}},this.#F=e=>{i.enteredPin==e.target&&(i.enteredPin=null,i.linkValid=!1,i.link?.setMessagePlaceNode())}}startDrag(e){this.link=new Ve(this.target,null),this.blueprint.linksContainerElement.prepend(this.link),this.link.setMessagePlaceNode(),this.#z=this.blueprint.querySelectorAll("ueb-pin"),this.#z.forEach((e=>{e!=this.target&&(e.getClickableElement().addEventListener("mouseenter",this.#B),e.getClickableElement().addEventListener("mouseleave",this.#F))})),this.link.startDragging(),this.link.setDestinationLocation(e)}dragTo(e,t){this.link.setDestinationLocation(e)}endDrag(){this.#z.forEach((e=>{e.removeEventListener("mouseenter",this.#B),e.removeEventListener("mouseleave",this.#F)})),this.enteredPin&&this.linkValid?(this.blueprint.addGraphElement(this.link),this.link.destinationPin=this.enteredPin,this.link.removeMessage(),this.link.finishDragging()):(this.link.finishDragging(),this.link.remove()),this.enteredPin=null,this.link=null,this.#z=null}}class We extends Ae{static styles=r``;connectedCallback(e){super.connectedCallback(e),e.nodeElement=e.closest("ueb-node")}createInputObjects(e){return[new Ke(e.clickableElement,e.blueprint,{moveEverywhere:!0,looseTarget:!0})]}render(e){const t=O`
${this.renderIcon(e)}
`,n=O`
${e.getPinDisplayName()} ${this.renderInput(e)}
`;return O`
${e.isInput()?O`${t}${n}`:O`${n}${t}`}
`}renderIcon(e){return O``}renderInput(e){return O``}firstUpdated(e,t){super.firstUpdated(e,t),e.dataset.id=e.GetPinIdValue(),e.clickableElement=e}getLinkLocation(e){const t=e.querySelector(".ueb-pin-icon").getBoundingClientRect(),n=ne.convertLocation([(t.left+t.right)/2,(t.top+t.bottom)/2],e.blueprint.gridElement);return e.blueprint.compensateTranslation(n)}}class Xe extends We{#U;get inputContentElements(){return this.#U}firstUpdated(e,t){if(super.firstUpdated(e,t),this.#U=[...e.querySelectorAll(".ueb-pin-input-content")],this.#U.length){this.setInputs(e,this.getInputs(e));let t=this;this.onFocusHandler=t=>e.blueprint.dispatchEditTextEvent(!0),this.onFocusOutHandler=n=>{n.preventDefault(),document.getSelection()?.removeAllRanges(),t.setInputs(e,this.getInputs(e)),e.blueprint.dispatchEditTextEvent(!1)},this.#U.forEach((e=>{e.addEventListener("focus",this.onFocusHandler),e.addEventListener("focusout",this.onFocusOutHandler)}))}}cleanup(e){super.cleanup(e),this.#U.forEach((e=>{e.removeEventListener("focus",this.onFocusHandler),e.removeEventListener("focusout",this.onFocusOutHandler)}))}createInputObjects(e){return[...super.createInputObjects(e),...this.#U.map((t=>new Fe(t,e.blueprint)))]}getInput(e){return this.getInputs(e).reduce(((e,t)=>e+t),"")}getInputs(e){return this.#U.map((e=>e.innerHTML.replaceAll("
","\n")))}setInputs(e,t=[],n=!0){this.#U.forEach(((e,n)=>e.innerText=t[n])),n&&e.setDefaultValue(t.reduce(((e,t)=>e+t),""))}renderInput(e){return e.isInput()?O`
`:O``}}class Ye extends Xe{#G;firstUpdated(e,t){super.firstUpdated(e,t),this.#G=e.querySelector(".ueb-pin-input");let n=this;this.onChangeHandler=t=>e.entity.DefaultValue=n.#G.checked?"true":"false",this.#G.addEventListener("change",this.onChangeHandler)}cleanup(e){super.cleanup(e),this.#G.removeEventListener("change",this.onChangeHandler)}getInputs(e){return[this.#G.checked?"true":"false"]}renderInput(e){return e.isInput()?O``:super.renderInput(e)}}class qe extends We{renderIcon(e){return O``}}class Ze extends Xe{#G;firstUpdated(e,t){super.firstUpdated(e,t),this.#G=e.querySelector(".ueb-pin-input")}getInputs(e){return[this.#G.dataset.linearColor]}setInputs(e,t=[]){}renderInput(e){return e.isInput()?O``:super.renderInput(e)}}class Je extends Xe{onInputHandler;firstUpdated(e,t){super.firstUpdated(e,t),this.onInputHandler=e=>{e.stopPropagation(),("insertParagraph"==e.inputType||"insertLineBreak"==e.inputType||"insertFromPaste"==e.inputType&&e.target.innerText.includes("\n"))&&(e.target.blur(),this.inputContentElements.forEach((e=>e.innerText=e.innerText.replaceAll("\n",""))))},this.inputContentElements.forEach((e=>{e.addEventListener("input",this.onInputHandler)}))}cleanup(e){super.cleanup(e),this.inputContentElements.forEach((e=>{e.removeEventListener("input",this.onInputHandler)}))}getInputs(e){return this.inputContentElements.map((e=>e.textContent))}setInputs(e,t=[],n=!0){t=t.map((e=>e.replaceAll("\n",""))),super.setInputs(e,t,n)}}class Qe extends Xe{setInputs(e,t=[]){let n=parseFloat(t.length?t[0]:this.getInput(e)),i=!0;isNaN(n)&&(n=parseFloat(""!=e.entity.DefaultValue?e.entity.DefaultValue:e.entity.AutogeneratedDefaultValue)),isNaN(n)&&(n=0,i=!1),t[0]=ne.minDecimals(n),super.setInputs(e,t,i)}}class et extends Xe{firstUpdated(e,t){super.firstUpdated(e,t)}}class tt extends Xe{#G;firstUpdated(e,t){super.firstUpdated(e,t),this.#G=e.querySelector(".ueb-pin-input")}getInputs(e){return[this.#G.dataset.linearColor]}setInputs(e,t=[]){}renderInput(e){return e.isInput()?O`
`:super.renderInput(e)}}class nt extends ze{static#R={bool:Ye,exec:qe,name:Je,real:Qe,string:et,"/Script/CoreUObject.LinearColor":Ze,"/Script/CoreUObject.Vector":tt};static properties={advancedView:{type:String,attribute:"data-advanced-view",reflect:!0},color:{type:he,converter:{fromAttribute:(e,t)=>e?xe.grammar.LinearColorFromAnyColor.parse(e).value:null,toAttribute:(e,t)=>e?ne.printLinearColor(e):null},attribute:"data-color",reflect:!0},defaultValue:{type:String,attribute:!1},isLinked:{type:Boolean,converter:ne.booleanConverter,attribute:"data-linked",reflect:!0},pinType:{type:String,attribute:"data-type",reflect:!0},pinDirection:{type:String,attribute:"data-direction",reflect:!0}};static getTypeTemplate(e){return nt.#R[e.getType()]??We}nodeElement;clickableElement;connections=0;get defaultValue(){return this.unreactiveDefaultValue}set defaultValue(e){let t=this.unreactiveDefaultValue;this.unreactiveDefaultValue=e,this.requestUpdate("defaultValue",t)}constructor(e){super(e,new(nt.getTypeTemplate(e))),this.advancedView=e.bAdvancedView,this.unreactiveDefaultValue=e.getDefaultValue(),this.pinType=this.entity.getType(),this.color=this.constructor.properties.color.converter.fromAttribute(J.pinColor[this.pinType]?.toString()),this.isLinked=!1,this.pinDirection=e.isInput()?"input":e.isOutput()?"output":"hidden",this.entity.subscribe("DefaultValue",(e=>this.defaultValue=e.toString())),this.entity.subscribe("PinToolTip",(e=>{let t=e.match(/\s*(.+?(?=\n)|.+\S)\s*/);return t?ne.formatStringName(t[1]):ne.formatStringName(this.entity.PinName)}))}connectedCallback(){super.connectedCallback()}GetPinId(){return this.entity.PinId}GetPinIdValue(){return this.GetPinId().value}getPinName(){return this.entity.PinName}getPinDisplayName(){let e=null;return this.entity.PinToolTip&&(e=this.entity.PinToolTip.match(/\s*(.+?(?=\n)|.+\S)\s*/))?ne.formatStringName(e[1]):ne.formatStringName(this.entity.PinName)}isInput(){return this.entity.isInput()}isOutput(){return this.entity.isOutput()}getClickableElement(){return this.clickableElement}getLinkLocation(){return this.template.getLinkLocation(this)}getNodeElement(){return this.nodeElement}getLinks(){return this.entity.LinkedTo??[]}setDefaultValue(e){this.entity.DefaultValue=e}sanitizeLinks(){this.entity.LinkedTo=this.getLinks().filter((e=>{let t=this.blueprint.getPin(e);if(t){this.blueprint.getLink(this,t,!0)||this.blueprint.addGraphElement(new Ve(this,t))}return t}))}linkTo(e){this.entity.linkTo(e.getNodeElement().getNodeName(),e.entity),this.isLinked=this.entity.isLinked()}unlinkFrom(e){this.entity.unlinkFrom(e.getNodeElement().getNodeName(),e.entity),this.isLinked=this.entity.isLinked()}redirectLink(e,t){const n=this.entity.LinkedTo.findIndex((t=>t.objectName.toString()==e.getNodeElement().getNodeName()&&t.pinGuid.valueOf()==e.entity.PinId.valueOf()));return n>=0&&(this.entity.LinkedTo[n]=t,!0)}}customElements.define("ueb-pin",nt);class it extends He{constructor(e,t,n){super(e,t,n),this.stepSize=parseInt(n?.stepSize??J.gridSize),this.mouseLocation=[0,0]}startDrag(){this.mouseLocation=ne.snapToGrid(this.clickedPosition,this.stepSize),this.target.selected||(this.blueprint.unselectAll(),this.target.setSelected(!0))}dragTo(e,t){const n=[this.target.locationX,this.target.locationY],[i,r]=this.stepSize>1?[ne.snapToGrid(e,this.stepSize),ne.snapToGrid(n,this.stepSize)]:[e,n],s=[i[0]-this.mouseLocation[0],i[1]-this.mouseLocation[1]];0==s[0]&&0==s[1]||(s[0]+=r[0]-this.target.locationX,s[1]+=r[1]-this.target.locationY,this.target.dispatchDragEvent(s),this.mouseLocation=i)}unclicked(){this.started||(this.blueprint.unselectAll(),this.target.setSelected(!0))}}class rt extends Ae{createInputObjects(e){return[...super.createInputObjects(e),new it(e,e.blueprint,{looseTarget:!0})]}update(e,t){super.update(e,t),t.has("locationX")&&e.style.setProperty("--ueb-position-x",`${e.locationX}`),t.has("locationY")&&e.style.setProperty("--ueb-position-y",`${e.locationY}`)}firstUpdated(e,t){super.firstUpdated(e,t),e.selected&&!e.listeningDrag&&e.setSelected(!0)}}class st extends rt{toggleAdvancedDisplayHandler;render(e){return O`
${e.nodeDisplayName}
${"DevelopmentOnly"==e.enabledState?.toString()?O`
Development Only
`:O``} ${e.advancedPinDisplay?O`
`:O``}
`}async firstUpdated(e,t){super.firstUpdated(e,t);const n=e.querySelector(".ueb-node-inputs"),i=e.querySelector(".ueb-node-outputs");Promise.all(e.getPinElements().map((e=>e.updateComplete))).then((()=>e.dispatchReflowEvent())),e.getPinElements().forEach((e=>{e.isInput()?n.appendChild(e):e.isOutput()&&i.appendChild(e)})),this.toggleAdvancedDisplayHandler=t=>{e.toggleShowAdvancedPinDisplay(),e.addNextUpdatedCallbacks((()=>e.dispatchReflowEvent()),!0)},e.nodeNameElement=e.querySelector(".ueb-node-name-text")}getPinElements(e){return e.querySelectorAll("ueb-pin")}}class ot extends Be{static properties={...Be.properties,name:{type:String,attribute:"data-name",reflect:!0},advancedPinDisplay:{type:String,attribute:"data-advanced-display",converter:le.attributeConverter,reflect:!0},enabledState:{type:String,attribute:"data-enabled-state",reflect:!0},nodeDisplayName:{type:String,attribute:!1},pureFunction:{type:Boolean,converter:ne.booleanConverter,attribute:"data-pure-function"}};get blueprint(){return super.blueprint}set blueprint(e){super.blueprint=e,this.#V.forEach((t=>t.blueprint=e))}#K;get nodeNameElement(){return this.#K}set nodeNameElement(e){this.#K=e}#V;constructor(e){super(e,new st),this.#V=this.getPinEntities().filter((e=>!e.isHidden())).map((e=>new nt(e))),this.#V.forEach((e=>e.nodeElement=this)),this.name=e.getObjectName(),this.advancedPinDisplay=e.AdvancedPinDisplay?.toString(),this.enabledState=e.EnabledState,this.nodeDisplayName=e.getDisplayName(),this.pureFunction=e.bIsPureFunc,this.dragLinkObjects=[],super.setLocation([this.entity.NodePosX.value,this.entity.NodePosY.value]),this.entity.subscribe("AdvancedPinDisplay",(e=>this.advancedPinDisplay=e)),this.entity.subscribe("Name",(e=>this.name=e))}static fromSerializedObject(e){e=e.trim();let t=ie.getSerializer(ye).deserialize(e);return new ot(t)}connectedCallback(){this.getAttribute("type")?.trim(),super.connectedCallback()}disconnectedCallback(){super.disconnectedCallback(),this.dispatchDeleteEvent()}getNodeName(){return this.entity.getObjectName()}getNodeDisplayName(){return this.entity.getDisplayName()}sanitizeLinks(){this.getPinElements().forEach((e=>e.sanitizeLinks()))}rename(e){if(this.entity.Name==e)return!1;for(let t of this.getPinElements())for(let n of t.getLinks())this.blueprint.getPin(n).redirectLink(t,new me({objectName:e,pinGuid:t.entity.PinId}));this.entity.Name=e}getPinElements(){return this.#V}getPinEntities(){return this.entity.CustomProperties.filter((e=>e instanceof ve))}setLocation(e=[0,0]){let t=this.entity.NodePosX.constructor;this.entity.NodePosX=new t(e[0]),this.entity.NodePosY=new t(e[1]),super.setLocation(e)}dispatchDeleteEvent(e){let t=new CustomEvent(J.nodeDeleteEventName,{bubbles:!0,cancelable:!0});this.dispatchEvent(t)}dispatchReflowEvent(){let e=new CustomEvent(J.nodeReflowEventName,{bubbles:!0,cancelable:!0});this.dispatchEvent(e)}setShowAdvancedPinDisplay(e){this.entity.AdvancedPinDisplay=new le(e?"Shown":"Hidden")}toggleShowAdvancedPinDisplay(){this.setShowAdvancedPinDisplay("Shown"!=this.entity.AdvancedPinDisplay?.toString())}}customElements.define("ueb-node",ot);class at extends Q{#W;constructor(e,t,n={}){n.listenOnFocus=!0,n.unlistenOnTextEdit=!0,super(e,t,n),this.serializer=new Ce;let i=this;this.#W=e=>i.pasted(e.clipboardData.getData("Text"))}listenEvents(){document.body.addEventListener("paste",this.#W)}unlistenEvents(){document.body.removeEventListener("paste",this.#W)}pasted(e){let t=0,n=0,i=0,r=this.serializer.readMultiple(e).map((e=>{let r=new ot(e);return t+=r.locationY,n+=r.locationX,++i,r}));t/=i,n/=i,r.length>0&&this.blueprint.unselectAll();let s=this.blueprint.mousePosition;return r.forEach((e=>{const i=[s[0]-n,s[1]-t];e.addLocation(i),e.snapToGrid(),e.setSelected(!0)})),this.blueprint.addGraphElement(...r),!0}}class lt extends He{constructor(e,t,n){super(e,t,n),this.selectorElement=this.blueprint.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 ut{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 ct{constructor(e,t,n,i){this.initialPosition=e,this.finalPosition=e,this.metadata=new Array(t.length),this.primaryOrder=new ut((e=>this.metadata[e].primaryBoundary)),this.secondaryOrder=new ut((e=>this.metadata[e].secondaryBoundary)),this.selectFunc=i,this.rectangles=t,this.primaryOrder.reserve(this.rectangles.length),this.secondaryOrder.reserve(this.rectangles.length),t.forEach(((e,t)=>{let r={primaryBoundary:this.initialPosition[0],secondaryBoundary:this.initialPosition[1],rectangle:t,onSecondaryAxis:!1};this.metadata[t]=r,i(e,!1);const s=n(e);this.initialPosition[1]{if(this.metadata[n].onSecondaryAxis)this.selectFunc(this.rectangles[n],i);else if(i){this.secondaryOrder.insert(n,e[1]);const i=this.metadata[n].secondaryBoundary;Math.sign(e[1]-i)==t[1]&&Math.sign(i-this.initialPosition[1])==t[1]&&this.selectFunc(this.rectangles[n],!0)}else this.selectFunc(this.rectangles[n],!1),this.secondaryOrder.remove(n);this.computeBoundaries(),this.selectTo(e)};e[0]this.boundaries.primaryN.v&&e[0]this.boundaries.primaryP.v&&(++this.primaryOrder.currentPosition,n(this.boundaries.primaryP.i,this.initialPosition[0]{this.selectFunc(this.rectangles[t],n),this.computeBoundaries(),this.selectTo(e)};e[1]this.boundaries.secondaryN.v&&e[1]this.boundaries.secondaryP.v&&(++this.secondaryOrder.currentPosition,i(this.boundaries.secondaryP.i,this.initialPosition[1]i.clickedSomewhere(e.target),this.blueprint.focus&&document.addEventListener("click",this.#X)}clickedSomewhere(e){e.closest("ueb-blueprint")||this.blueprint.setFocused(!1)}listenEvents(){document.addEventListener("click",this.#X)}unlistenEvents(){document.removeEventListener("click",this.#X)}}class gt extends Ae{static styleVariables={"--ueb-font-size":`${J.fontSize}`,"--ueb-grid-axis-line-color":`${J.gridAxisLineColor}`,"--ueb-grid-expand":`${J.expandGridSize}px`,"--ueb-grid-line-color":`${J.gridLineColor}`,"--ueb-grid-line-width":`${J.gridLineWidth}px`,"--ueb-grid-set-line-color":`${J.gridSetLineColor}`,"--ueb-grid-set":`${J.gridSet}`,"--ueb-grid-size":`${J.gridSize}px`,"--ueb-link-min-width":`${J.linkMinWidth}`,"--ueb-node-radius":`${J.nodeRadius}px`,"--ueb-pin-bool-color":`${J.pinColor.bool}`,"--ueb-pin-default-color":`${J.pinColor.default}`,"--ueb-pin-exec-color":`${J.pinColor.exec}`,"--ueb-pin-name-color":`${J.pinColor.name}`,"--ueb-pin-real-color":`${J.pinColor.real}`,"--ueb-pin-string-color":`${J.pinColor.string}`,"--ueb-pin-linear-color":`${J.pinColor["/Script/CoreUObject.LinearColor"]}`};constructed(e){e.style.cssText=Object.entries(gt.styleVariables).map((([e,t])=>`${e}:${t};`)).join("")}createInputObjects(e){return[new Ne(e.getGridDOMElement(),e),new at(e.getGridDOMElement(),e),new Le(e.getGridDOMElement(),e),new _e(e.getGridDOMElement(),e),new De(e.getGridDOMElement(),e,{looseTarget:!0}),new lt(e.getGridDOMElement(),e,{clickButton:0,exitAnyButton:!0,looseTarget:!0,moveEverywhere:!0}),new Ie(e.getGridDOMElement(),e,{clickButton:2,exitAnyButton:!1,looseTarget:!0,moveEverywhere:!0}),new pt(e.getGridDOMElement(),e),new je(e.getGridDOMElement(),e),new Me(e.getGridDOMElement(),e)]}render(e){return O`
1:1
`}firstUpdated(e,t){super.firstUpdated(e,t),e.headerElement=e.querySelector(".ueb-viewport-header"),e.overlayElement=e.querySelector(".ueb-viewport-overlay"),e.viewportElement=e.querySelector(".ueb-viewport-body"),e.selectorElement=new ht,e.querySelector(".ueb-grid-content")?.append(e.selectorElement),e.gridElement=e.viewportElement.querySelector(".ueb-grid"),e.linksContainerElement=e.querySelector("[data-links]"),e.linksContainerElement.append(...e.getLinks()),e.nodesContainerElement=e.querySelector("[data-nodes]"),e.nodesContainerElement.append(...e.getNodes()),e.viewportElement.scroll(J.expandGridSize,J.expandGridSize)}updated(e,t){super.updated(e,t),(t.has("scrollX")||t.has("scrollY"))&&e.viewportElement.scroll(e.scrollX,e.scrollY)}getPin(e,t){return e.querySelector(`ueb-node[data-name="${t.objectName}"] ueb-pin[data-id="${t.pinGuid}"]`)}}class mt extends ze{static properties={selecting:{type:Boolean,attribute:"data-selecting",reflect:!0,converter:ne.booleanConverter},scrolling:{type:Boolean,attribute:"data-scrolling",reflect:!0,converter:ne.booleanConverter},focused:{type:Boolean,attribute:"data-focused",reflect:!0,converter:ne.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 styles=gt.styles;#Y=new Map;nodes=[];links=[];mousePosition=[0,0];gridElement;viewportElement;overlayElement;selectorElement;linksContainerElement;nodesContainerElement;headerElement;focused=!1;nodeBoundariesSupplier=e=>{let t=e.getBoundingClientRect(),n=this.nodesContainerElement.getBoundingClientRect();const i=1/this.getScale();return{primaryInf:(t.left-n.left)*i,primarySup:(t.right-n.right)*i,secondaryInf:(t.top-n.top)*i,secondarySup:(t.bottom-n.bottom)*i}};nodeSelectToggleFunction=(e,t)=>{e.setSelected(t)};constructor(e=new J){super({},new gt),this.selecting=!1,this.scrolling=!1,this.focused=!1,this.zoom=0,this.scrollX=J.expandGridSize,this.scrollY=J.expandGridSize,this.translateX=J.expandGridSize,this.translateY=J.expandGridSize}getGridDOMElement(){return this.gridElement}disconnectedCallback(){super.disconnectedCallback()}getScroll(){return[this.scrollX,this.scrollY]}setScroll([e,t],n=!1){this.scrollX=e,this.scrollY=t}scrollDelta(e,t=!1){const n=[2*J.expandGridSize,2*J.expandGridSize];let i=this.getScroll(),r=[i[0]+e[0],i[1]+e[1]],s=[0,0];for(let t=0;t<2;++t)e[t]<0&&r[t]0&&r[t]>n[t]-J.gridExpandThreshold*J.expandGridSize&&(s[t]=1);0==s[0]&&0==s[1]||this.seamlessExpand(s),i=this.getScroll(),r=[i[0]+e[0],i[1]+e[1]],this.setScroll(r,t)}scrollCenter(){const e=this.getScroll(),t=[this.translateX-e[0],this.translateY-e[1]],n=this.getViewportSize().map((e=>e/2)),i=[t[0]-n[0],t[1]-n[1]];this.scrollDelta(i,!0)}getViewportSize(){return[this.viewportElement.clientWidth,this.viewportElement.clientHeight]}getScrollMax(){return[this.viewportElement.scrollWidth-this.viewportElement.clientWidth,this.viewportElement.scrollHeight-this.viewportElement.clientHeight]}snapToGrid(e){return ne.snapToGrid(e,J.gridSize)}seamlessExpand([e,t]){e=Math.round(e),t=Math.round(t);let n=this.getScale();[e,t]=[-e*J.expandGridSize,-t*J.expandGridSize],0!=e&&(this.scrollX+=e,e/=n),0!=t&&(this.scrollY+=t,t/=n),this.translateX+=e,this.translateY+=t}progressiveSnapToGrid(e){return J.expandGridSize*Math.round(e/J.expandGridSize+.5*Math.sign(e))}getZoom(){return this.zoom}setZoom(e,t){if((e=ne.clamp(e,J.minZoom,J.maxZoom))==this.zoom)return;let n=this.getScale();this.zoom=e,t&&requestAnimationFrame((e=>{t[0]+=this.translateX,t[1]+=this.translateY;let i=this.getScale()/n,r=[i*t[0],i*t[1]];this.scrollDelta([(r[0]-t[0])*n,(r[1]-t[1])*n])}))}getScale(){return parseFloat(getComputedStyle(this.gridElement).getPropertyValue("--ueb-scale"))}compensateTranslation([e,t]){return[e-=this.translateX,t-=this.translateY]}getNodes(e=!1){return e?this.nodes.filter((e=>e.selected)):this.nodes}getPin(e){return[...this.nodes.find((t=>e.objectName.toString()==t.getNodeName()))?.getPinElements()??[]].find((t=>e.pinGuid.toString()==t.GetPinIdValue()))}getLinks([e,t]=[]){if(null==e!=t==null){const n=e??t;return this.links.filter((e=>e.sourcePin==n||e.destinationPin==n))}return null!=e&&null!=t?this.links.filter((n=>n.sourcePin==e&&n.destinationPin==t||n.sourcePin==t&&n.destinationPin==e)):this.links}getLink(e,t,n=!1){return this.links.find((i=>i.sourcePin==e&&i.destinationPin==t||n&&i.sourcePin==t&&i.destinationPin==e))}selectAll(){this.getNodes().forEach((e=>this.nodeSelectToggleFunction(e,!0)))}unselectAll(){this.getNodes().forEach((e=>this.nodeSelectToggleFunction(e,!1)))}addGraphElement(...e){for(let t of e)if(t.blueprint=this,t instanceof ot&&!this.nodes.includes(t)){const e=t.entity.getObjectName(),n=this.nodes.find((t=>t.entity.getObjectName()==e));if(n){let e=n.entity.getObjectName(!0);this.#Y[e]=this.#Y[e]??-1;do{++this.#Y[e]}while(this.nodes.find((t=>t.entity.getObjectName()==J.nodeName(e,this.#Y[e]))));n.rename(J.nodeName(e,this.#Y[e]))}this.nodes.push(t),this.nodesContainerElement?.appendChild(t)}else t instanceof Ve&&!this.links.includes(t)&&(this.links.push(t),this.linksContainerElement&&!this.linksContainerElement.contains(t)&&this.linksContainerElement.appendChild(t));e.filter((e=>e instanceof ot)).forEach((e=>e.sanitizeLinks()))}removeGraphElement(...e){for(let t of e)if(t.closest("ueb-blueprint")==this){t.remove();let e=t instanceof ot?this.nodes:t instanceof Ve?this.links:null;e?.splice(e.findIndex((e=>e===t)),1)}}setFocused(e=!0){if(this.focused==e)return;let t=new CustomEvent(e?"blueprint-focus":"blueprint-unfocus");this.focused=e,this.focused||this.unselectAll(),this.dispatchEvent(t)}dispatchEditTextEvent(e){const t=new CustomEvent(e?J.editTextEventName.begin:J.editTextEventName.end);this.dispatchEvent(t)}}customElements.define("ueb-blueprint",mt);class ft extends xe{constructor(e,t,n,i,r,s,o){e=e??(e=>`(${e})`),super(t,n,i,r,s,o),this.wrap=e}read(e){const t=ke.getGrammarForType(xe.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,n=!1){return this.wrap(this.subWrite(e,[],t,n))}}class vt extends ft{#q;constructor(e,t){super(void 0,t),this.#q=e}write(e,t,n=!1){return this.#q(t,n)}}class bt extends ft{constructor(e){super(void 0,e)}write(e,t,n){return n||t.constructor!==String?ne.encodeString(t.toString()):`"${ne.encodeString(t.toString())}"`}}!function(){const e=e=>`(${e})`;ie.registerSerializer(null,new vt(((e,t)=>"()"),null)),ie.registerSerializer(Array,new vt(((e,t)=>`(${e.map((e=>ie.getSerializer(ne.getType(e)).serialize(e,t)+",")).join("")})`),Array)),ie.registerSerializer(Boolean,new vt(((e,t)=>e?t?"true":"True":t?"false":"False"),Boolean)),ie.registerSerializer(oe,new ft(e,oe)),ie.registerSerializer(ae,new bt(ae)),ie.registerSerializer(le,new bt(le)),ie.registerSerializer(ue,new bt(ue)),ie.registerSerializer(ce,new ft((e=>`${ce.lookbehind}(${e})`),ce,"",", ",!1,"",(e=>""))),ie.registerSerializer(de,new ft(e,de)),ie.registerSerializer(he,new ft(e,he)),ie.registerSerializer(pe,new ft((e=>`${pe.lookbehind}(${e})`),pe,"",", ",!1,"",(e=>""))),ie.registerSerializer(Number,new vt((e=>e.toString()),Number)),ie.registerSerializer(ye,new Ce),ie.registerSerializer(se,new vt((e=>(e.type??"")+(e.path?e.type?`'"${e.path}"'`:`"${e.path}"`:"")),se)),ie.registerSerializer(ge,new bt(ge)),ie.registerSerializer(ve,new ft((e=>`${ve.lookbehind} (${e})`),ve,"",",",!0)),ie.registerSerializer(me,new ft((e=>e),me,""," ",!1,"",(e=>""))),ie.registerSerializer(String,new vt(((e,t)=>t?ne.encodeString(e):`"${ne.encodeString(e)}"`),String))}();export{mt as Blueprint,J as Configuration,Ve as LinkElement,ot as NodeElement}; diff --git a/js/Utility.js b/js/Utility.js index f8b8feb..6ff82c6 100755 --- a/js/Utility.js +++ b/js/Utility.js @@ -1,7 +1,9 @@ +import CalculatedType from "./entity/CalculatedType" import TypeInitialization from "./entity/TypeInitialization" /** * @typedef {import("./entity/LinearColorEntity").default} LinearColorEntity + * @typedef {import("./entity/IEntity").default} IEntity */ export default class Utility { @@ -60,6 +62,26 @@ export default class Utility { return location } + /** + * @param {IEntity} + * @param {Object} target Object holding the data + * @param {String[]} keys The chained keys to access from object in order to get the value + * @param {Boolean} defaultValue Value to return in case from doesn't have it + * @returns {any} The value in from corresponding to the keys or defaultValue otherwise + */ + static isSerialized(entity, keys, propertyDefinition = Utility.objectGet(entity.constructor.attributes, keys)) { + if (propertyDefinition instanceof CalculatedType) { + return Utility.isSerialized(entity, keys, propertyDefinition.calculate(entity)) + } + if (propertyDefinition instanceof TypeInitialization) { + if (propertyDefinition.serialized) { + return true + } + return Utility.isSerialized(entity, keys, propertyDefinition.type) + } + return false + } + /** * Gets a value from an object, gives defaultValue in case of failure * @param {Object} target Object holding the data @@ -128,6 +150,9 @@ export default class Utility { } static getType(value) { + if (value === null) { + return null + } let constructor = value?.constructor switch (constructor) { case TypeInitialization: @@ -207,8 +232,9 @@ export default class Utility { */ static encodeString(value, input = false) { return value + .replaceAll('"', '\\"') // Escape " + .replaceAll("\n", "\\n") // Replace newline with \n .replaceAll("\u00A0", " ") // Replace special space symbol - .replaceAll("\n", String.raw`\n`) // Replace newline with \n } /** @@ -216,8 +242,9 @@ export default class Utility { */ static decodeString(value, input = false) { return value - .replaceAll(" ", "\u00A0") // Replace special space symbol - .replaceAll(String.raw`\n`, "\n") // Replace newline with \n + .replaceAll('\\"', '"') + .replaceAll("\\n", "\n") + .replaceAll(" ", "\u00A0") } /** diff --git a/js/entity/CalculatedType.js b/js/entity/CalculatedType.js index 9f7eefd..fcd0a46 100644 --- a/js/entity/CalculatedType.js +++ b/js/entity/CalculatedType.js @@ -17,6 +17,6 @@ export default class CalculatedType { * @param {IEntity} entity */ calculate(entity) { - return this.f(entity) + return this.#f(entity) } } \ No newline at end of file diff --git a/js/entity/IEntity.js b/js/entity/IEntity.js index 6e1eb1a..dc8e330 100644 --- a/js/entity/IEntity.js +++ b/js/entity/IEntity.js @@ -1,12 +1,12 @@ +import CalculatedType from "./CalculatedType" import Observable from "../Observable" +import SerializerFactory from "../serialization/SerializerFactory" import TypeInitialization from "./TypeInitialization" import Utility from "../Utility" -import CalculatedType from "./CalculatedType" export default class IEntity extends Observable { static attributes = {} - #showAsString = false constructor(values) { super() @@ -21,19 +21,25 @@ export default class IEntity extends Observable { Object.getOwnPropertyNames(properties), Object.getOwnPropertyNames(values ?? {}) )) { - + let value = Utility.objectGet(values, [property]) let defaultValue = properties[property] - const defaultType = Utility.getType(defaultValue) + let defaultType = Utility.getType(defaultValue) + if (defaultValue instanceof CalculatedType) { + defaultValue = defaultValue.calculate(this) + defaultType = Utility.getType(defaultValue) + } if (!(property in properties)) { - console.warn(`Property ${prefix}${property} is not defined in ${this.constructor.name}.attributes`) + console.warn( + `Property ${prefix}${property} in the serialized data is not defined in ${this.constructor.name}.properties` + ) } else if ( - defaultValue != null + !(property in values) + && defaultValue !== undefined && !(defaultValue instanceof TypeInitialization && !defaultValue.showDefault) - && !(property in values) ) { console.warn( - `${this.constructor.name} adds property ${prefix}${property} not defined in the serialized data` + `${this.constructor.name}.properties will add property ${prefix}${property} not defined in the serialized data` ) } @@ -44,34 +50,34 @@ export default class IEntity extends Observable { continue } - /* - * The value can either be: - * - Array: can contain multiple values, its property is assigned multiple times like (X=1, X="4"). - * - CalculatedType: the exact type depends on the previous attributes assigned to this entity. - * - TypeInitialization: contains the maximum amount of information about the attribute. - * - A type: the default value will be default constructed object without arguments. - * - A proper value. - */ - const value = Utility.objectGet(values, [property]) if (value !== undefined) { - target[property] = TypeInitialization.sanitize(value, defaultType) - // We have a value, need nothing more - continue - } - if (defaultValue instanceof CalculatedType) { - defaultValue = defaultValue.calculate(this) - defaultType = Utility.getType(defaultValue) + // Remember value can still be null + if ( + value?.constructor === String + && defaultValue instanceof TypeInitialization + && defaultValue.serialized + && defaultValue.type !== String + ) { + value = SerializerFactory.getSerializer(defaultValue.type).deserialize(value) + } + target[property] = TypeInitialization.sanitize(value, Utility.getType(defaultValue)) + continue // We have a value, need nothing more } + if (defaultValue instanceof TypeInitialization) { if (!defaultValue.showDefault) { target[property] = undefined // Declare undefined to preserve the order of attributes continue } - defaultValue = defaultValue.value + if (defaultValue.serialized) { + defaultValue = "" + } else { + defaultType = defaultValue.type + defaultValue = defaultValue.value + } } if (defaultValue instanceof Array) { - target[property] = [] - continue + defaultValue = [] } target[property] = TypeInitialization.sanitize(defaultValue, defaultType) } @@ -85,15 +91,4 @@ export default class IEntity extends Observable { } defineAllAttributes(this, attributes, values) } - - isShownAsString() { - return this.#showAsString - } - - /** - * @param {Boolean} v - */ - setShowAsString(v) { - this.#showAsString = v - } } diff --git a/js/entity/PinEntity.js b/js/entity/PinEntity.js index c9a3aa9..ae33606 100755 --- a/js/entity/PinEntity.js +++ b/js/entity/PinEntity.js @@ -1,3 +1,4 @@ +import CalculatedType from "./CalculatedType" import GuidEntity from "./GuidEntity" import IEntity from "./IEntity" import LinearColorEntity from "./LinearColorEntity" @@ -6,9 +7,22 @@ import ObjectReferenceEntity from "./ObjectReferenceEntity" import PinReferenceEntity from "./PinReferenceEntity" import SerializedType from "./SerializedType" import TypeInitialization from "./TypeInitialization" +import VectorEntity from "./VectorEntity" export default class PinEntity extends IEntity { + static #typeEntityMap = { + "/Script/CoreUObject.LinearColor": LinearColorEntity, + "/Script/CoreUObject.Vector": VectorEntity, + "bool": Boolean, + "exec": String, + "name": String, + "real": Number, + "string": String, + } + static get typeEntityMap() { + return PinEntity.#typeEntityMap + } static lookbehind = "Pin" static attributes = { PinId: GuidEntity, @@ -30,12 +44,16 @@ export default class PinEntity extends IEntity { bSerializeAsSinglePrecisionFloat: false, }, LinkedTo: new TypeInitialization([PinReferenceEntity], false), - DefaultValue: new TypeInitialization( - new SerializedType([ - LinearColorEntity, - ]), - false - ), + DefaultValue: + new CalculatedType( + /** @param {PinEntity} pinEntity */ + pinEntity => new TypeInitialization( + PinEntity.typeEntityMap[pinEntity.getType()] ?? String, + false, + undefined, + true + ) + ), AutogeneratedDefaultValue: new TypeInitialization(String, false), DefaultObject: new TypeInitialization(ObjectReferenceEntity, false, null), PersistentGuid: GuidEntity, @@ -47,6 +65,13 @@ export default class PinEntity extends IEntity { bOrphanedPin: false, } + getType() { + if (this.PinType.PinCategory == "struct") { + return this.PinType.PinSubCategoryObject.path + } + return this.PinType.PinCategory + } + getDefaultValue() { return this.DefaultValue ?? "" } @@ -108,13 +133,6 @@ export default class PinEntity extends IEntity { return false } - getType() { - if (this.PinType.PinCategory == "struct") { - return this.PinType.PinSubCategoryObject.path - } - return this.PinType.PinCategory - } - getSubCategory() { return this.PinType.PinSubCategoryObject.path } diff --git a/js/entity/TypeInitialization.js b/js/entity/TypeInitialization.js index eef0621..f1fa217 100755 --- a/js/entity/TypeInitialization.js +++ b/js/entity/TypeInitialization.js @@ -1,5 +1,3 @@ -import SerializedType from "./SerializedType" - /** * @template T */ @@ -31,13 +29,22 @@ export default class TypeInitialization { this.#value = v } + /** @type {Boolean} */ + #serialized + get serialized() { + return this.#serialized + } + set serialized(v) { + this.#serialized = v + } + static sanitize(value, targetType) { if (targetType === undefined) { targetType = value?.constructor } if ( targetType - && targetType !== SerializedType + // value is not of type targetType && !(value?.constructor === targetType || value instanceof targetType) ) { value = new targetType(value) @@ -49,22 +56,25 @@ export default class TypeInitialization { } /** - * @typedef {(new () => T) | SerializedType | StringConstructor | NumberConstructor | BooleanConstructor} Constructor + * @typedef {(new () => T) | StringConstructor | NumberConstructor | BooleanConstructor} Constructor * @param {Constructor|Array} type * @param {Boolean} showDefault * @param {any} value + * @param {Boolean} serialized */ - constructor(type, showDefault = true, value = undefined) { + constructor(type, showDefault = true, value = undefined, serialized = false) { if (value === undefined) { if (type instanceof Array) { value = [] - } else if (type instanceof SerializedType) { + } else if (serialized) { value = "" } else { value = TypeInitialization.sanitize(new type()) } } - this.#showDefault = showDefault this.#type = type + this.#showDefault = showDefault + this.#value = value + this.#serialized = serialized } } diff --git a/js/entity/VectorEntity.js b/js/entity/VectorEntity.js index 810baa4..2fda0cc 100644 --- a/js/entity/VectorEntity.js +++ b/js/entity/VectorEntity.js @@ -1,6 +1,6 @@ import IEntity from "./IEntity" -export default class LinearColorEntity extends IEntity { +export default class VectorEntity extends IEntity { static attributes = { X: Number, diff --git a/js/serialization/CustomSerializer.js b/js/serialization/CustomSerializer.js index 5cd753b..632358b 100755 --- a/js/serialization/CustomSerializer.js +++ b/js/serialization/CustomSerializer.js @@ -5,16 +5,19 @@ import GeneralSerializer from "./GeneralSerializer" */ /** - * @template {IEntity} T + * @template {IEntity | Boolean | Number | String} T */ export default class CustomSerializer extends GeneralSerializer { + #objectWriter + /** + * @param {(v: T, insideString: Boolean) => String} objectWriter * @param {new () => T} entityType */ constructor(objectWriter, entityType) { super(undefined, entityType) - this.objectWriter = objectWriter + this.#objectWriter = objectWriter } /** @@ -22,8 +25,8 @@ export default class CustomSerializer extends GeneralSerializer { * @param {Boolean} insideString * @returns {String} */ - write(object, insideString = false) { - let result = this.objectWriter(object, insideString) + write(entity, object, insideString = false) { + let result = this.#objectWriter(object, insideString) return result } } diff --git a/js/serialization/GeneralSerializer.js b/js/serialization/GeneralSerializer.js index 0e61edb..b154a5d 100755 --- a/js/serialization/GeneralSerializer.js +++ b/js/serialization/GeneralSerializer.js @@ -37,8 +37,8 @@ export default class GeneralSerializer extends ISerializer { * @param {Boolean} insideString * @returns {String} */ - write(object, insideString = false) { - let result = this.wrap(this.subWrite([], object, insideString)) + write(entity, object, insideString = false) { + let result = this.wrap(this.subWrite(entity, [], object, insideString)) return result } } diff --git a/js/serialization/Grammar.js b/js/serialization/Grammar.js index 6d37446..cd489ea 100755 --- a/js/serialization/Grammar.js +++ b/js/serialization/Grammar.js @@ -16,7 +16,6 @@ import SerializedType from "../entity/SerializedType" import TypeInitialization from "../entity/TypeInitialization" import Utility from "../Utility" import VectorEntity from "../entity/VectorEntity" -import CalculatedType from "../entity/CalculatedType" /** * @typedef {import("../entity/IEntity").default} IEntity @@ -30,32 +29,9 @@ export default class Grammar { static getGrammarForType(r, attributeType, defaultGrammar) { if (attributeType instanceof TypeInitialization) { - // Unpack TypeInitialization - attributeType = attributeType.type - return Grammar.getGrammarForType(r, attributeType, defaultGrammar) - } - if (attributeType instanceof SerializedType) { - const nonStringTypes = attributeType.types.filter(t => t !== String) - let result = P.alt( - ...nonStringTypes.map(t => - Grammar.getGrammarForType(r, t).wrap(P.string('"'), P.string('"')).map( - /** - * @param {IEntity} entity - */ - entity => { - entity.setShowAsString(true) // Showing as string because it is inside a SerializedType - return entity - } - ) - ) - ) - if (nonStringTypes.length < attributeType.types.length) { - result = result.or(r.String/*.map(v => { - if (attributeType.stringFallback) { - console.log("Unrecognized value, fallback on String") - } - return v - })*/) // Separated because it cannot be wrapped into " and " + let result = Grammar.getGrammarForType(r, attributeType.type, defaultGrammar) + if (attributeType.serialized && !(attributeType.type instanceof String)) { + result = result.wrap(P.string('"'), P.string('"')) } return result } @@ -149,7 +125,12 @@ export default class Grammar { Null = r => P.seq(P.string("("), r.InlineOptWhitespace, P.string(")")).map(_ => null).desc("null: ()") - Boolean = r => P.alt(P.string("True"), P.string("False")).map(v => v === "True" ? true : false) + Boolean = r => P.alt( + P.string("True"), + P.string("true"), + P.string("False"), + P.string("false"), + ).map(v => v.toLocaleLowerCase() === "true" ? true : false) .desc("either True or False") HexDigit = r => P.regex(/[0-9a-fA-f]/).desc("hexadecimal digit") diff --git a/js/serialization/ISerializer.js b/js/serialization/ISerializer.js index b100345..6ef9fe0 100644 --- a/js/serialization/ISerializer.js +++ b/js/serialization/ISerializer.js @@ -34,13 +34,8 @@ export default class ISerializer { * @param {Boolean} insideString * @returns {String} */ - serialize(object, insideString) { - insideString ||= object.isShownAsString() - let result = this.write(object, insideString) - if (object.isShownAsString()) { - result = `"${result}"` - } - return result + serialize(object, insideString, entity = object) { + return this.write(entity, object, insideString) } /** @@ -56,7 +51,7 @@ export default class ISerializer { * @param {Boolean} insideString * @returns {String} */ - write(object, insideString) { + write(entity, object, insideString) { throw new Error("Not implemented") } @@ -64,30 +59,12 @@ export default class ISerializer { * @param {String[]} fullKey * @param {Boolean} insideString */ - writeValue(value, fullKey, insideString) { - if (value === null) { - return "()" - } - const serialize = v => SerializerFactory.getSerializer(Utility.getType(v)).serialize(v) - // This is an exact match (and not instanceof) to hit also primitive types (by accessing value.constructor they are converted to objects automatically) - switch (value?.constructor) { - case Function: - return this.writeValue(value(), fullKey, insideString) - case Boolean: - return Utility.firstCapital(value.toString()) - case Number: - return value.toString() - case String: - return insideString - ? `\\"${Utility.encodeString(value)}\\"` - : `"${Utility.encodeString(value)}"` - } - if (value instanceof Array) { - return `(${value.map(v => serialize(v) + ",").join("")})` - } - if (value instanceof IEntity) { - return serialize(value) + writeValue(entity, value, fullKey, insideString) { + const serializer = SerializerFactory.getSerializer(Utility.getType(value)) + if (!serializer) { + throw new Error("Unknown value type, a serializer must be registered in the SerializerFactory class") } + return serializer.write(entity, value, insideString) } /** @@ -96,7 +73,7 @@ export default class ISerializer { * @param {Boolean} insideString * @returns {String} */ - subWrite(key, object, insideString) { + subWrite(entity, key, object, insideString) { let result = "" let fullKey = key.concat("") const last = fullKey.length - 1 @@ -106,13 +83,18 @@ export default class ISerializer { if (value?.constructor === Object) { // Recursive call when finding an object result += (result.length ? this.separator : "") - + this.subWrite(fullKey, value, insideString) - } else if (value !== undefined && this.showProperty(object, fullKey, value)) { + + 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.separator : "") + this.prefix + this.attributeKeyPrinter(fullKey) + this.attributeValueConjunctionSign - + this.writeValue(value, fullKey, insideString) + + ( + isSerialized + ? `"${this.writeValue(entity, value, fullKey, true)}"` + : this.writeValue(entity, value, fullKey, insideString) + ) } } if (this.trailingSeparator && result.length && fullKey.length === 1) { @@ -122,7 +104,7 @@ export default class ISerializer { return result } - showProperty(object, attributeKey, attributeValue) { + showProperty(entity, object, attributeKey, attributeValue) { const attributes = this.entityType.attributes const attribute = Utility.objectGet(attributes, attributeKey) if (attribute instanceof TypeInitialization) { diff --git a/js/serialization/ObjectSerializer.js b/js/serialization/ObjectSerializer.js index 0729817..7514707 100755 --- a/js/serialization/ObjectSerializer.js +++ b/js/serialization/ObjectSerializer.js @@ -9,7 +9,7 @@ export default class ObjectSerializer extends ISerializer { super(ObjectEntity, " ", "\n", false) } - showProperty(object, attributeKey, attributeValue) { + showProperty(entity, object, attributeKey, attributeValue) { switch (attributeKey.toString()) { case "Class": case "Name": @@ -17,7 +17,7 @@ export default class ObjectSerializer extends ISerializer { // Serielized separately return false } - return super.showProperty(object, attributeKey, attributeValue) + return super.showProperty(entity, object, attributeKey, attributeValue) } /** @@ -46,9 +46,9 @@ export default class ObjectSerializer extends ISerializer { * @param {ObjectEntity} object * @param {Boolean} insideString */ - write(object, insideString) { - let result = `Begin Object Class=${object.Class.path} Name=${this.writeValue(object.Name, ["Name"], insideString)} -${this.subWrite([], object, insideString) + write(entity, object, insideString) { + let result = `Begin Object Class=${object.Class.path} Name=${this.writeValue(entity, object.Name, ["Name"], insideString)} +${this.subWrite(entity, [], object, insideString) + object .CustomProperties.map(pin => this.separator diff --git a/js/serialization/PinSerializer.js b/js/serialization/PinSerializer.js deleted file mode 100755 index 3503734..0000000 --- a/js/serialization/PinSerializer.js +++ /dev/null @@ -1,21 +0,0 @@ -import PinEntity from "../entity/PinEntity" -import Utility from "../Utility" -import GeneralSerializer from "./GeneralSerializer" - -export default class PinSerializer extends GeneralSerializer { - - constructor() { - super(v => `${PinEntity.lookbehind} (${v})`, PinEntity, "", ",", true) - } - - /** - * @param {String[]} fullKey - * @param {Boolean} insideString - */ - writeValue(value, fullKey, insideString) { - if (value?.constructor === String && fullKey.length == 1 && fullKey[0] == "DefaultValue") { - return `"${Utility.encodeInputString(value)}"` - } - return super.writeValue(value, fullKey, insideString) - } -} diff --git a/js/serialization/SerializerFactory.js b/js/serialization/SerializerFactory.js index 78a4319..111516a 100755 --- a/js/serialization/SerializerFactory.js +++ b/js/serialization/SerializerFactory.js @@ -1,14 +1,25 @@ import Utility from "../Utility" +/** @typedef {import("../entity/IEntity").default} IEntity */ +/** + * @template {IEntity} T + * @typedef {import("./ISerializer").default} ISerializer + */ + export default class SerializerFactory { + /** @type {Map>} */ static #serializers = new Map() static registerSerializer(entity, object) { SerializerFactory.#serializers.set(entity, object) } + /** + * @template {IEntity} T + * @param {T} entity + */ static getSerializer(entity) { - return SerializerFactory.#serializers.get(Utility.getType(entity)) + return SerializerFactory.#serializers.get(entity) } } diff --git a/js/serialization/ToStringSerializer.js b/js/serialization/ToStringSerializer.js index 32aad8d..4537013 100755 --- a/js/serialization/ToStringSerializer.js +++ b/js/serialization/ToStringSerializer.js @@ -1,3 +1,4 @@ +import Utility from "../Utility" import GeneralSerializer from "./GeneralSerializer" /** @@ -20,10 +21,9 @@ export default class ToStringSerializer extends GeneralSerializer { * @param {T} object * @param {Boolean} insideString */ - write(object, insideString) { - let result = insideString || object.isShownAsString() - ? `"${object.toString().replaceAll('"', '\\"')}"` - : object.toString() - return result + write(entity, object, insideString) { + return !insideString && object.constructor === String + ? `"${Utility.encodeString(object.toString())}"` // String will have quotes if not inside a string already + : Utility.encodeString(object.toString()) } } diff --git a/js/serialization/initializeSerializerFactory.js b/js/serialization/initializeSerializerFactory.js index c74495b..66d07e3 100755 --- a/js/serialization/initializeSerializerFactory.js +++ b/js/serialization/initializeSerializerFactory.js @@ -14,14 +14,52 @@ import ObjectSerializer from "./ObjectSerializer" import PathSymbolEntity from "../entity/PathSymbolEntity" import PinEntity from "../entity/PinEntity" import PinReferenceEntity from "../entity/PinReferenceEntity" -import PinSerializer from "./PinSerializer" import SerializerFactory from "./SerializerFactory" import ToStringSerializer from "./ToStringSerializer" +import Utility from "../Utility" export default function initializeSerializerFactory() { const bracketsWrapped = v => `(${v})` + SerializerFactory.registerSerializer( + null, + new CustomSerializer( + (nullValue, insideString) => "()", + null + ) + ) + + SerializerFactory.registerSerializer( + Array, + new CustomSerializer( + /** @param {Array} array */ + (array, insideString) => + `(${array + .map(v => + SerializerFactory.getSerializer(Utility.getType(v)).serialize(v, insideString) + "," + ) + .join("") + })`, + Array + ) + ) + + SerializerFactory.registerSerializer( + Boolean, + new CustomSerializer( + /** @param {Boolean} boolean */ + (boolean, insideString) => boolean + ? insideString + ? "true" + : "True" + : insideString + ? "false" + : "False", + Boolean + ) + ) + SerializerFactory.registerSerializer( FunctionReferenceEntity, new GeneralSerializer(bracketsWrapped, FunctionReferenceEntity) @@ -53,6 +91,15 @@ export default function initializeSerializerFactory() { new GeneralSerializer(v => `${LocalizedTextEntity.lookbehind}(${v})`, LocalizedTextEntity, "", ", ", false, "", _ => "") ) + SerializerFactory.registerSerializer( + Number, + new CustomSerializer( + /** @param {Number} value */ + value => value.toString(), + Number + ) + ) + SerializerFactory.registerSerializer( ObjectEntity, new ObjectSerializer() @@ -73,13 +120,23 @@ export default function initializeSerializerFactory() { SerializerFactory.registerSerializer(PathSymbolEntity, new ToStringSerializer(PathSymbolEntity)) + SerializerFactory.registerSerializer( + PinEntity, + new GeneralSerializer(v => `${PinEntity.lookbehind} (${v})`, PinEntity, "", ",", true) + ) + SerializerFactory.registerSerializer( PinReferenceEntity, new GeneralSerializer(v => v, PinReferenceEntity, "", " ", false, "", _ => "") ) SerializerFactory.registerSerializer( - PinEntity, - new PinSerializer() + String, + new CustomSerializer( + (value, insideString) => insideString + ? Utility.encodeString(value) + : `"${Utility.encodeString(value)}"`, + String + ) ) } diff --git a/js/template/IInputPinTemplate.js b/js/template/IInputPinTemplate.js index 5df2f24..57277fd 100644 --- a/js/template/IInputPinTemplate.js +++ b/js/template/IInputPinTemplate.js @@ -71,7 +71,6 @@ export default class IInputPinTemplate extends PinTemplate { * @param {PinElement} pin */ getInputs(pin) { - Utility.decodeInputString(pin.entity.DefaultValue) return this.#inputContentElements.map(element => // Faster than innerText which causes reflow element.innerHTML.replaceAll("
", "\n"))