nodePort.ts 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. import { BlockTools } from '../blockTools';
  2. import { NodeMaterialBlockConnectionPointTypes } from 'babylonjs/Materials/Node/Enums/nodeMaterialBlockConnectionPointTypes';
  3. import { NodeMaterialConnectionPoint } from 'babylonjs/Materials/Node/nodeMaterialBlockConnectionPoint';
  4. import { GlobalState } from '../globalState';
  5. import { Nullable } from 'babylonjs/types';
  6. import { Observer } from 'babylonjs/Misc/observable';
  7. import { Vector2 } from 'babylonjs/Maths/math.vector';
  8. import { IDisplayManager } from './display/displayManager';
  9. import { GraphNode } from './graphNode';
  10. import { NodeLink } from './nodeLink';
  11. import { GraphFrame } from './graphFrame';
  12. import { FrameNodePort } from './frameNodePort';
  13. import { FramePortData } from './graphCanvas';
  14. export class NodePort {
  15. protected _element: HTMLDivElement;
  16. protected _img: HTMLImageElement;
  17. protected _globalState: GlobalState;
  18. protected _portLabelElement: Element;
  19. protected _onCandidateLinkMovedObserver: Nullable<Observer<Nullable<Vector2>>>;
  20. protected _onSelectionChangedObserver: Nullable<Observer<Nullable<GraphFrame | GraphNode | NodeLink | NodePort | FramePortData>>>;
  21. protected _exposedOnFrame: boolean;
  22. public isExposed = false;
  23. public delegatedPort: Nullable<FrameNodePort> = null;
  24. public get element(): HTMLDivElement {
  25. if (this.delegatedPort) {
  26. return this.delegatedPort.element;
  27. }
  28. return this._element;
  29. }
  30. public get portName(){
  31. let portName = this.connectionPoint.displayName || this.connectionPoint.name;
  32. if (this.connectionPoint.ownerBlock.isInput) {
  33. portName = this.node.name;
  34. }
  35. return portName
  36. }
  37. public set portName(newName: string){
  38. if(this._portLabelElement) {
  39. this.connectionPoint.displayName = newName;
  40. this._portLabelElement.innerHTML = newName;
  41. }
  42. }
  43. public get disabled() {
  44. if (!this.connectionPoint.isConnected) {
  45. return false;
  46. } else if (this._isConnectedToNodeOutsideOfFrame()) { //connected to outside node
  47. return true;
  48. } else {
  49. const link = this.node.getLinksForConnectionPoint(this.connectionPoint)
  50. if (link.length){
  51. if (link[0].nodeB === this.node) { // check if this node is the receiving
  52. return true;
  53. }
  54. }
  55. }
  56. return false;
  57. }
  58. public hasLabel(){
  59. return !!this._portLabelElement;
  60. }
  61. public get exposedOnFrame() {
  62. if(!!this.connectionPoint.isExposedOnFrame || this._isConnectedToNodeOutsideOfFrame()) {
  63. return true
  64. } return false;
  65. }
  66. public set exposedOnFrame(value: boolean) {
  67. if(this.disabled){
  68. return;
  69. }
  70. this.connectionPoint.isExposedOnFrame = value;
  71. }
  72. public get exposedPortPosition()
  73. {
  74. return this.connectionPoint.exposedPortPosition;
  75. }
  76. public set exposedPortPosition(value: number) {
  77. this.connectionPoint.exposedPortPosition = value;
  78. }
  79. private _isConnectedToNodeOutsideOfFrame() {
  80. const link = this.node.getLinksForConnectionPoint(this.connectionPoint)
  81. if (link.length){
  82. for(let i = 0; i < link.length; i++){
  83. if (link[i].nodeA.enclosingFrameId !== link[i].nodeB!.enclosingFrameId) {
  84. return true;
  85. }
  86. }
  87. }
  88. return false;
  89. }
  90. public refresh() {
  91. this._element.style.background = BlockTools.GetColorFromConnectionNodeType(this.connectionPoint.type);
  92. switch (this.connectionPoint.type) {
  93. case NodeMaterialBlockConnectionPointTypes.Float:
  94. case NodeMaterialBlockConnectionPointTypes.Int:
  95. this._img.src = "data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMSAyMSI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5WZWN0b3IxPC90aXRsZT48ZyBpZD0iTGF5ZXJfNSIgZGF0YS1uYW1lPSJMYXllciA1Ij48Y2lyY2xlIGNsYXNzPSJjbHMtMSIgY3g9IjEwLjUiIGN5PSIxMC41IiByPSI3LjUiLz48L2c+PC9zdmc+";
  96. break;
  97. case NodeMaterialBlockConnectionPointTypes.Vector2:
  98. this._img.src = "data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMSAyMSI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5WZWN0b3IyPC90aXRsZT48ZyBpZD0iTGF5ZXJfNSIgZGF0YS1uYW1lPSJMYXllciA1Ij48cGF0aCBjbGFzcz0iY2xzLTEiIGQ9Ik0zLDEwLjVhNy41Miw3LjUyLDAsMCwwLDYuNSw3LjQzVjMuMDdBNy41Miw3LjUyLDAsMCwwLDMsMTAuNVoiLz48cGF0aCBjbGFzcz0iY2xzLTEiIGQ9Ik0xMS41LDMuMDdWMTcuOTNhNy41LDcuNSwwLDAsMCwwLTE0Ljg2WiIvPjwvZz48L3N2Zz4=";
  99. break;
  100. case NodeMaterialBlockConnectionPointTypes.Vector3:
  101. case NodeMaterialBlockConnectionPointTypes.Color3:
  102. this._img.src = "data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMSAyMSI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5WZWN0b3IzPC90aXRsZT48ZyBpZD0iTGF5ZXJfNSIgZGF0YS1uYW1lPSJMYXllciA1Ij48cGF0aCBjbGFzcz0iY2xzLTEiIGQ9Ik0zLjU3LDEzLjMxLDkuNSw5Ljg5VjNBNy41MSw3LjUxLDAsMCwwLDMsMTAuNDYsNy4zMiw3LjMyLDAsMCwwLDMuNTcsMTMuMzFaIi8+PHBhdGggY2xhc3M9ImNscy0xIiBkPSJNMTYuNDMsMTUsMTAuNSwxMS42Miw0LjU3LDE1YTcuNDgsNy40OCwwLDAsMCwxMS44NiwwWiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTE4LDEwLjQ2QTcuNTEsNy41MSwwLDAsMCwxMS41LDNWOS44OWw1LjkzLDMuNDJBNy4zMiw3LjMyLDAsMCwwLDE4LDEwLjQ2WiIvPjwvZz48L3N2Zz4=";
  103. break;
  104. case NodeMaterialBlockConnectionPointTypes.Vector4:
  105. case NodeMaterialBlockConnectionPointTypes.Color4:
  106. this._img.src = "data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMSAyMSI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5WZWN0b3I0PC90aXRsZT48ZyBpZD0iTGF5ZXJfNSIgZGF0YS1uYW1lPSJMYXllciA1Ij48cGF0aCBjbGFzcz0iY2xzLTEiIGQ9Ik0xMS41LDExLjV2Ni40M2E3LjUxLDcuNTEsMCwwLDAsNi40My02LjQzWiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTExLjUsMy4wN1Y5LjVoNi40M0E3LjUxLDcuNTEsMCwwLDAsMTEuNSwzLjA3WiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTkuNSwxNy45M1YxMS41SDMuMDdBNy41MSw3LjUxLDAsMCwwLDkuNSwxNy45M1oiLz48cGF0aCBjbGFzcz0iY2xzLTEiIGQ9Ik05LjUsMy4wN0E3LjUxLDcuNTEsMCwwLDAsMy4wNyw5LjVIOS41WiIvPjwvZz48L3N2Zz4=";
  107. break;
  108. case NodeMaterialBlockConnectionPointTypes.Matrix:
  109. this._img.src = "data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMSAyMSI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5NYXRyaXg8L3RpdGxlPjxnIGlkPSJMYXllcl81IiBkYXRhLW5hbWU9IkxheWVyIDUiPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTExLjUsNi4xMVY5LjVoMy4zOUE0LjUxLDQuNTEsMCwwLDAsMTEuNSw2LjExWiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTExLjUsMTQuODlhNC41MSw0LjUxLDAsMCwwLDMuMzktMy4zOUgxMS41WiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTExLjUsMy4wN3YyQTUuNTQsNS41NCwwLDAsMSwxNS45Miw5LjVoMkE3LjUxLDcuNTEsMCwwLDAsMTEuNSwzLjA3WiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTE1LjkyLDExLjVhNS41NCw1LjU0LDAsMCwxLTQuNDIsNC40MnYyYTcuNTEsNy41MSwwLDAsMCw2LjQzLTYuNDNaIi8+PHBhdGggY2xhc3M9ImNscy0xIiBkPSJNNS4wOCwxMS41aC0yQTcuNTEsNy41MSwwLDAsMCw5LjUsMTcuOTN2LTJBNS41NCw1LjU0LDAsMCwxLDUuMDgsMTEuNVoiLz48cGF0aCBjbGFzcz0iY2xzLTEiIGQ9Ik05LjUsMy4wN0E3LjUxLDcuNTEsMCwwLDAsMy4wNyw5LjVoMkE1LjU0LDUuNTQsMCwwLDEsOS41LDUuMDhaIi8+PHBhdGggY2xhc3M9ImNscy0xIiBkPSJNOS41LDExLjVINi4xMUE0LjUxLDQuNTEsMCwwLDAsOS41LDE0Ljg5WiIvPjxwYXRoIGNsYXNzPSJjbHMtMSIgZD0iTTkuNSw2LjExQTQuNTEsNC41MSwwLDAsMCw2LjExLDkuNUg5LjVaIi8+PC9nPjwvc3ZnPg==";
  110. break;
  111. case NodeMaterialBlockConnectionPointTypes.Object:
  112. this._img.src = "data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCAyMSAyMSI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5WZWN0b3IxPC90aXRsZT48ZyBpZD0iTGF5ZXJfNSIgZGF0YS1uYW1lPSJMYXllciA1Ij48Y2lyY2xlIGNsYXNzPSJjbHMtMSIgY3g9IjEwLjUiIGN5PSIxMC41IiByPSI3LjUiLz48L2c+PC9zdmc+";
  113. break;
  114. }
  115. }
  116. public constructor(portContainer: HTMLElement, public connectionPoint: NodeMaterialConnectionPoint, public node: GraphNode, globalState: GlobalState) {
  117. this._element = portContainer.ownerDocument!.createElement("div");
  118. this._element.classList.add("port");
  119. portContainer.appendChild(this._element);
  120. this._globalState = globalState;
  121. this._img = portContainer.ownerDocument!.createElement("img");
  122. this._element.appendChild(this._img );
  123. // determine if node name is editable
  124. if (portContainer.children[0].className === 'port-label') {
  125. this._portLabelElement = portContainer.children[0];
  126. }
  127. (this._element as any).port = this;
  128. // Drag support
  129. this._element.ondragstart= () => false;
  130. this._onCandidateLinkMovedObserver = globalState.onCandidateLinkMoved.add(coords => {
  131. const rect = this._element.getBoundingClientRect();
  132. if (!coords || rect.left > coords.x || rect.right < coords.x || rect.top > coords.y || rect.bottom < coords.y) {
  133. this._element.classList.remove("selected");
  134. return;
  135. }
  136. this._element.classList.add("selected");
  137. this._globalState.onCandidatePortSelectedObservable.notifyObservers(this);
  138. });
  139. this._onSelectionChangedObserver = this._globalState.onSelectionChangedObservable.add((selection) => {
  140. if (selection === this) {
  141. this._img.classList.add("selected");
  142. } else {
  143. this._img.classList.remove("selected");
  144. }
  145. });
  146. this.refresh();
  147. }
  148. public dispose() {
  149. this._globalState.onCandidateLinkMoved.remove(this._onCandidateLinkMovedObserver);
  150. if (this._onSelectionChangedObserver) {
  151. this._globalState.onSelectionChangedObservable.remove(this._onSelectionChangedObserver);
  152. }
  153. }
  154. public static CreatePortElement(connectionPoint: NodeMaterialConnectionPoint, node: GraphNode, root: HTMLElement,
  155. displayManager: Nullable<IDisplayManager>, globalState: GlobalState) {
  156. let portContainer = root.ownerDocument!.createElement("div");
  157. let block = connectionPoint.ownerBlock;
  158. portContainer.classList.add("portLine");
  159. root.appendChild(portContainer);
  160. if (!displayManager || displayManager.shouldDisplayPortLabels(block)) {
  161. let portLabel = root.ownerDocument!.createElement("div");
  162. portLabel.classList.add("port-label");
  163. portLabel.innerHTML = connectionPoint.displayName || connectionPoint.name;
  164. portContainer.appendChild(portLabel);
  165. }
  166. return new NodePort(portContainer, connectionPoint, node, globalState);
  167. }
  168. }