Source: pathComponent.mjs

  1. import { Component } from "./component.mjs";
  2. import { Pin } from "./pin.mjs";
  3. import { Coordinate } from "./coordinate.mjs";
  4. import { Net } from "./net.mjs";
  5. /**
  6. * Class representing a path-style TikZ component.
  7. *
  8. * @extends Component
  9. */
  10. class PathComponent extends Component {
  11. /**
  12. * @property {string} tikzComponentName - the tikz component name, e.g. "R"
  13. * @private
  14. */
  15. #tikzComponentName;
  16. /** @property {string} instanceName - the instance name, e.g. "R1" */
  17. instanceName;
  18. /** @property {Pin[]} pins - a list of the two corresponding pins */
  19. pins;
  20. /**
  21. * Generate a PathComponent or -stencil.
  22. *
  23. * @param {string} tikzComponentName - the tikz component name, e.g. "R"
  24. * @param {string} [instanceName=""] - the instance name, e.g. "R1"
  25. * @param {Pin[]} pins - a list of the two corresponding pins
  26. * @param {number} [angle=0] - the angle to rotate the component
  27. * @param {boolean} [mirrorX=false] - true to mirror on x axis
  28. * @param {boolean} [mirrorY=false] - true to mirror on y axis
  29. */
  30. constructor(tikzComponentName, instanceName, pins, angle = 0, mirrorX = false, mirrorY = false) {
  31. super(angle || 0, mirrorX, mirrorY);
  32. this.#tikzComponentName = tikzComponentName;
  33. this.instanceName = instanceName || "";
  34. this.pins = pins;
  35. }
  36. /**
  37. * Getts the private tikzComponentName.
  38. *
  39. * @returns {string}
  40. */
  41. get tikzComponentName() {
  42. return this.#tikzComponentName;
  43. }
  44. /**
  45. * Serializes a component. The TikZ "source code" is returned.
  46. *
  47. * @param {number} [indent=0] - the indention (= amount of tabs) to use
  48. * @returns {string} the serialized component
  49. */
  50. serialize(indent = 0) {
  51. // C1 --> ${C}_{1}$
  52. let label = "";
  53. if (this.instanceName) {
  54. let [_fullMatch, name, index] = this.instanceName.match(/^([a-zA-Z]+)[_-]?([0-9]+)$/) || [null, null, null];
  55. if (name && !Number.isNaN((index = Number.parseInt(index)))) label = `\${${name}}_{${index}}\$`;
  56. else label = this.instanceName.replace("_", "\\_");
  57. label = ", l=" + label;
  58. }
  59. /** DEBUG: marks the first pin of the component. (1)-/--[R]---(2) */
  60. /** @type {string} */
  61. let pinmark;
  62. if (global.DEBUG) {
  63. pinmark = "\t".repeat(indent) + "\\draw[color=blue] " + this.pins[0].coord.serializeName() + " ";
  64. if (this.angle === 0 || this.angle === 180) {
  65. if (this.pins[0].coord.x < this.pins[1].coord.x)
  66. // -->
  67. pinmark += "++ (3pt,0) ++ (-1.25pt, -2.5pt) -- ++(2.5pt, 5pt);\n";
  68. // <--
  69. else pinmark += "++ (-3pt,0) ++ (-1.25pt, -2.5pt) -- ++(2.5pt, 5pt);\n";
  70. } else if (this.angle === 90 || this.angle === -90) {
  71. if (this.pins[0].coord.y < this.pins[1].coord.y)
  72. // up
  73. pinmark += "++ (0,3pt) ++ (-2.5pt, -1.25pt) -- ++(5pt, 2.5pt);\n";
  74. // down
  75. else pinmark += "++ (0,-3pt) ++ (-2.5pt, -1.25pt) -- ++(5pt, 2.5pt);\n";
  76. } else pinmark = ""; // unknown / diagonal
  77. } else pinmark = "";
  78. return (
  79. pinmark +
  80. "\t".repeat(indent) +
  81. (global.DEBUG ? "\\draw[color=blue] " : "\\draw ") +
  82. this.pins[0].coord.serializeName() +
  83. " to[" +
  84. this.#tikzComponentName +
  85. label +
  86. "] " +
  87. this.pins[1].coord.serializeName() +
  88. ";"
  89. );
  90. }
  91. /**
  92. * Deep clone of this object.
  93. *
  94. * @returns {PathComponent} the cloned object
  95. */
  96. deepClone() {
  97. return new PathComponent(this.#tikzComponentName, this.instanceName, this.pins.deepClone(), this.angle);
  98. }
  99. /**
  100. * Generate a TikZ component using `this` as an stencil. The parameters are informations extracted from ABL.
  101. *
  102. * @param {string} libraryName - the ABL library name, e.g. "ads_rflib"
  103. * @param {string} cellName - the librarys component name, e.g. "R"
  104. * @param {string} instanceName - the instance/component name, e.g. "R1"
  105. * @param {Map<string,string>} attributes - map of all (XML) attributes
  106. * @param {Pin[]} pins - the pins with their name and number (position not yet set)
  107. * @param {{x: number, y: number, angle: number, xScale: number, yScale: number, mirrorX: boolean, mirrorY: boolean, scaling: number}} placement - general component placement information
  108. * @param {Wire[]} wires - list of all wires
  109. * @param {Map<string,Net>} nets - list of all nets
  110. * @param {Coordinate[]} coords - list of all coordinates
  111. *
  112. * @returns {PathComponent} the new PathComponent
  113. */
  114. useAsStencil(libraryName, cellName, instanceName, attributes, pins, placement, wires, nets, coords) {
  115. // normalize angle --> -180 < angle <= 180
  116. while (placement.angle <= -180) placement.angle += 360;
  117. while (placement.angle > 180) placement.angle -= 360;
  118. // get coordinate
  119. let instanceCoord = new Coordinate(placement.x * placement.scaling, placement.y * placement.scaling);
  120. const oldCoord = coords.find((existingCoord) => instanceCoord.equals(existingCoord));
  121. if (oldCoord) instanceCoord = oldCoord;
  122. else coords.push(instanceCoord);
  123. // reorder
  124. pins = this.pins.map((ablPin) =>
  125. pins.find((pin) =>
  126. pin.name && ablPin.name ? pin.name == ablPin.name : pin.instTermNumber === ablPin.instTermNumber
  127. )
  128. );
  129. // set pins (+positions)
  130. pins.forEach((pin) => pin.findPosition(this.pins, placement, instanceCoord, wires, nets, coords));
  131. return new PathComponent(this.#tikzComponentName, instanceName, pins, placement.angle);
  132. }
  133. }
  134. export { PathComponent };