import Configuration from "../Configuration.js" import pinColor from "../decoding/pinColor.js" import pinTitle from "../decoding/pinTitle.js" import Grammar from "../serialization/Grammar.js" import AlternativesEntity from "./AlternativesEntity.js" import ArrayEntity from "./ArrayEntity.js" import BooleanEntity from "./BooleanEntity.js" import ByteEntity from "./ByteEntity.js" import ComputedTypeEntity from "./ComputedTypeEntity.js" import EnumDisplayValueEntity from "./EnumDisplayValueEntity.js" import EnumEntity from "./EnumEntity.js" import FormatTextEntity from "./FormatTextEntity.js" import GuidEntity from "./GuidEntity.js" import IEntity from "./IEntity.js" import Integer64Entity from "./Integer64Entity.js" import IntegerEntity from "./IntegerEntity.js" import InvariantTextEntity from "./InvariantTextEntity.js" import LinearColorEntity from "./LinearColorEntity.js" import LocalizedTextEntity from "./LocalizedTextEntity.js" import NumberEntity from "./NumberEntity.js" import ObjectReferenceEntity from "./ObjectReferenceEntity.js" import PinReferenceEntity from "./PinReferenceEntity.js" import PinTypeEntity from "./PinTypeEntity.js" import RBSerializationVector2DEntity from "./RBSerializationVector2DEntity.js" import RotatorEntity from "./RotatorEntity.js" import SimpleSerializationRotatorEntity from "./SimpleSerializationRotatorEntity.js" import SimpleSerializationVector2DEntity from "./SimpleSerializationVector2DEntity.js" import SimpleSerializationVector4DEntity from "./SimpleSerializationVector4DEntity.js" import SimpleSerializationVectorEntity from "./SimpleSerializationVectorEntity.js" import StringEntity from "./StringEntity.js" import Vector2DEntity from "./Vector2DEntity.js" import Vector4DEntity from "./Vector4DEntity.js" import VectorEntity from "./VectorEntity.js" /** @template {IEntity} T */ export default class PinEntity extends IEntity { static lookbehind = "INVTEXT" static #typeEntityMap = { [Configuration.paths.linearColor]: LinearColorEntity, [Configuration.paths.rotator]: RotatorEntity, [Configuration.paths.vector]: VectorEntity, [Configuration.paths.vector2D]: Vector2DEntity, [Configuration.paths.vector4f]: Vector4DEntity, "bool": BooleanEntity, "byte": ByteEntity, "enum": EnumEntity, "exec": StringEntity, "int": IntegerEntity, "int64": Integer64Entity, "name": StringEntity, "real": NumberEntity, "string": StringEntity, } static #alternativeTypeEntityMap = { "enum": EnumDisplayValueEntity, "rg": RBSerializationVector2DEntity, [Configuration.paths.rotator]: SimpleSerializationRotatorEntity, [Configuration.paths.vector]: SimpleSerializationVectorEntity, [Configuration.paths.vector2D]: SimpleSerializationVector2DEntity, [Configuration.paths.vector3f]: SimpleSerializationVectorEntity, [Configuration.paths.vector4f]: SimpleSerializationVector4DEntity, } static attributes = { ...super.attributes, PinId: GuidEntity.withDefault(), PinName: StringEntity.withDefault(), PinFriendlyName: AlternativesEntity.accepting( LocalizedTextEntity, FormatTextEntity, InvariantTextEntity, StringEntity ), PinToolTip: StringEntity, Direction: StringEntity, PinType: PinTypeEntity.withDefault().flagInlined(), LinkedTo: ArrayEntity.of(PinReferenceEntity), SubPins: ArrayEntity.of(PinReferenceEntity), ParentPin: PinReferenceEntity, DefaultValue: ComputedTypeEntity.from( /** @param {PinEntity} pinEntity */ pinEntity => pinEntity.getEntityType(true) ?? StringEntity ).flagSerialized(), AutogeneratedDefaultValue: StringEntity, DefaultObject: ObjectReferenceEntity, PersistentGuid: GuidEntity, bHidden: BooleanEntity.withDefault(), bNotConnectable: BooleanEntity.withDefault(), bDefaultValueIsReadOnly: BooleanEntity.withDefault(), bDefaultValueIsIgnored: BooleanEntity.withDefault(), bAdvancedView: BooleanEntity.withDefault(), bOrphanedPin: BooleanEntity.withDefault(), } static grammar = Grammar.createEntityGrammar(this) #recomputesNodeTitleOnChange = false set recomputesNodeTitleOnChange(value) { this.#recomputesNodeTitleOnChange = value } get recomputesNodeTitleOnChange() { return this.#recomputesNodeTitleOnChange } #objectEntity get objectEntity() { return this.#objectEntity } set objectEntity(value) { this.#objectEntity = value } #pinIndex get pinIndex() { return this.#pinIndex } set pinIndex(value) { this.#pinIndex = value } constructor(values = {}) { super(values) /** @type {InstanceType} */ this.PinId /** @type {InstanceType} */ this.PinName /** @type {InstanceType} */ this.PinFriendlyName /** @type {InstanceType} */ this.PinToolTip /** @type {InstanceType} */ this.Direction /** @type {InstanceType} */ this.PinType /** @type {InstanceType} */ this.LinkedTo /** @type {InstanceType} */ this.DefaultValue /** @type {InstanceType} */ this.AutogeneratedDefaultValue /** @type {InstanceType} */ this.DefaultObject /** @type {InstanceType} */ this.PersistentGuid /** @type {InstanceType} */ this.bHidden /** @type {InstanceType} */ this.bNotConnectable /** @type {InstanceType} */ this.bDefaultValueIsReadOnly /** @type {InstanceType} */ this.bDefaultValueIsIgnored /** @type {InstanceType} */ this.bAdvancedView /** @type {InstanceType} */ this.bOrphanedPin } /** @param {ObjectEntity} objectEntity */ static fromLegacyObject(objectEntity) { return new PinEntity(objectEntity) } getType() { const category = this.PinType.PinCategory.toLocaleLowerCase() if (category === "struct" || category === "class" || category === "object" || category === "type") { return this.PinType.PinSubCategoryObject.path } if (this.isEnum()) { return "enum" } if (this.objectEntity?.isPcg()) { const pcgSuboject = this.objectEntity.getPcgSubobject() const pinObjectReference = this.isInput() ? pcgSuboject.InputPins?.[this.pinIndex] : pcgSuboject.OutputPins?.[this.pinIndex] if (pinObjectReference) { /** @type {ObjectEntity} */ const pinObject = pcgSuboject[Configuration.subObjectAttributeNameFromReference(pinObjectReference, true)] let allowedTypes = pinObject.Properties?.AllowedTypes?.toString() ?? "" if (allowedTypes == "") { allowedTypes = this.PinType.PinCategory ?? "" if (allowedTypes == "") { allowedTypes = "Any" } } if (allowedTypes) { if ( pinObject.Properties.bAllowMultipleData !== false && pinObject.Properties.bAllowMultipleConnections !== false ) { allowedTypes += "[]" } return allowedTypes } } } if (category === "optional") { switch (this.PinType.PinSubCategory) { case "red": return "real" case "rg": return "rg" case "rgb": return Configuration.paths.vector case "rgba": return Configuration.paths.linearColor default: return this.PinType.PinSubCategory } } return category } getEntityType(alternative = false) { const typeString = this.getType() const entity = PinEntity.#typeEntityMap[typeString] const alternativeEntity = PinEntity.#alternativeTypeEntityMap[typeString] return alternative && alternativeEntity !== undefined ? alternativeEntity : entity } pinTitle() { return pinTitle(this) } /** @param {PinEntity} other */ copyTypeFrom(other) { this.PinType.PinCategory = other.PinType.PinCategory this.PinType.PinSubCategory = other.PinType.PinSubCategory this.PinType.PinSubCategoryObject = other.PinType.PinSubCategoryObject this.PinType.PinSubCategoryMemberReference = other.PinType.PinSubCategoryMemberReference this.PinType.PinValueType = other.PinType.PinValueType this.PinType.ContainerType = other.PinType.ContainerType this.PinType.bIsReference = other.PinType.bIsReference this.PinType.bIsConst = other.PinType.bIsConst this.PinType.bIsWeakPointer = other.PinType.bIsWeakPointer this.PinType.bIsUObjectWrapper = other.PinType.bIsUObjectWrapper this.PinType.bSerializeAsSinglePrecisionFloat = other.PinType.bSerializeAsSinglePrecisionFloat } getDefaultValue(maybeCreate = false) { if (this.DefaultValue === undefined && maybeCreate) { this.DefaultValue = new (this.getEntityType(true))() } return this.DefaultValue } isEnum() { const type = this.PinType.PinSubCategoryObject.type return type === Configuration.paths.enum || type === Configuration.paths.userDefinedEnum || type.toLowerCase() === "enum" } isExecution() { return this.PinType.PinCategory === "exec" } isHidden() { return this.bHidden } isInput() { return !this.bHidden && this.Direction != "EGPD_Output" } isOutput() { return !this.bHidden && this.Direction == "EGPD_Output" } isLinked() { return this.LinkedTo?.length > 0 ?? false } /** * @param {String} targetObjectName * @param {PinEntity} targetPinEntity * @returns true if it was not already linked to the tarket */ linkTo(targetObjectName, targetPinEntity) { const linkFound = this.LinkedTo?.some(pinReferenceEntity => pinReferenceEntity.objectName.toString() == targetObjectName && pinReferenceEntity.pinGuid.valueOf() == targetPinEntity.PinId.valueOf() ) if (!linkFound) { (this.LinkedTo ??= []).push(new PinReferenceEntity(targetObjectName, targetPinEntity.PinId,)) return true } return false // Already linked } /** * @param {String} targetObjectName * @param {PinEntity} targetPinEntity * @returns true if it was linked to the target */ unlinkFrom(targetObjectName, targetPinEntity) { const indexElement = this.LinkedTo?.findIndex(pinReferenceEntity => { return pinReferenceEntity.objectName.toString() == targetObjectName && pinReferenceEntity.pinGuid.valueOf() == targetPinEntity.PinId.valueOf() }) if (indexElement >= 0) { this.LinkedTo.splice(indexElement, 1) if (this.LinkedTo.length === 0 && PinEntity.attributes.LinkedTo.default === undefined) { this.LinkedTo = undefined } return true } return false } getSubCategory() { return this.PinType.PinSubCategoryObject.path } pinColor() { return pinColor(this) } }