Source: ablTransistorMapper.mjs

import { Coordinate } from "./coordinate.mjs";
import { Pin } from "./pin.mjs";
import { Transistor } from "./transistor.mjs";

/**
 * Class for mapping and geometrical transforming of ABL transistors to TikZ. It contains position information both for
 * ABL and TikZ.
 *
 * @property {Transistor} tikzTransistor - the transistor used as stencil
 * @property {Pin[]} pins - top, bottom and tap pin (ABL)
 * @property {0 | 1 | 2} anchorNr - the selected anchor; 0=top, 1=bottom, 2=tap (ABL)
 */
class ABLTransistorMapper {
	tikzTransistor;
	pins;
	anchorNr;

	/**
	 * Generate the "converter" for an existing TikZ transistor stencil.
	 *
	 * @param {Transistor} tikzTransistor - the transistor used as stencil
	 * @param {Pin[]} pins - top, bottom and tap coordinate
	 * @param {0 | 1 | 2} anchorNr - the selected anchor; 0=top, 1=bottom, 2=tap
	 */
	constructor(tikzTransistor, pins, anchorNr) {
		this.tikzTransistor = tikzTransistor;
		this.pins = pins;
		this.anchorNr = anchorNr;
	}

	/**
	 * Calculates the coordinate where the top-bottom line crosses the orthogonal tap line. This can but does not need
	 * to be the center point.
	 *
	 * @returns {Coordinate}
	 */
	get lineCrossingCoord() {
		return this.pins[2].coord.orthogonalProjection(this.pins[0].coord, this.pins[1].coord);
	}

	/**
	 * Mirror coords/pins around the anchor.
	 *
	 * @param {boolean} mirrorX - true to mirror
	 * @param {boolean} mirrorY - true to mirror
	 */
	mirror(mirrorX, mirrorY) {
		if (!(mirrorX || mirrorY)) return;

		// Update mirror flags (xor)
		this.mirrorX = this.mirrorX != mirrorX;
		this.mirrorY = this.mirrorY != mirrorY;

		const mirrorMid = this.pins[this.anchorNr].coord;

		for (let i = 0; i < this.pins.length; i++) {
			if (i !== this.anchorCoord) {
				const mirrorCoord = this.pins[i].coord;
				mirrorCoord.subtract(mirrorMid);
				if (mirrorX) mirrorCoord.y = -mirrorCoord.y;

				if (mirrorY) mirrorCoord.x = -mirrorCoord.x;

				mirrorCoord.add(mirrorMid);
			}
		}
	}

	/**
	 * Rotate every coord around the anchor. The rotation is counter clockwise, like the default mathematical rotation.
	 *
	 * @param {number} angle
	 */
	rotate(angle) {
		if (!Number.isFinite(angle) || angle === 0) return;

		for (let i = 0; i < this.pins.length; i++) {
			if (i !== this.anchorNr) this.pins[i].coord.rotate(angle);
		}
	}

	/**
	 * Move all pins/coordinates using a vector.
	 *
	 * @param {Coordinate} vector - vector to add to all coordinates
	 */
	translate(vector) {
		this.pins.forEach((pin) => pin.coord.add(vector));
	}

	/**
	 * Clones this instance. This is neither a shallow nor a deep clone. Only the pins are deeply cloned.
	 *
	 * @returns {ABLTransistorMapper} - the semi deep clone
	 */
	clone() {
		return new ABLTransistorMapper(
			this.tikzTransistor,
			this.pins.map((pin) => pin.deepClone()),
			this.anchorNr
		);
	}

	/**
	 * Generate a TikZ component using `this` as an stencil. The parameters are informations extracted from ABL.
	 *
	 * @param {string} libraryName - the ABL library name, e.g. "ads_rflib"
	 * @param {string} cellName - the librarys component name, e.g. "R"
	 * @param {string} instanceName - the instance/component name, e.g. "R1"
	 * @param {Map<string,string>} attributes - map of all (XML) attributes
	 * @param {Pin[]} pins - the pins with their name and number (position not yet set)
	 * @param {{x: number, y: number, angle: number, xScale: number, yScale: number, mirrorX: boolean, mirrorY: boolean, scaling: number}} placement - general component placement information
	 * @param {Wire[]} wires - list of all wires
	 * @param {Map<string,Net>} nets - list of all nets
	 * @param {Coordinate[]} coords - list of all coordinates
	 *
	 * @returns {Transistor} the new Transistor
	 */
	useAsStencil(libraryName, cellName, instanceName, attributes, pins, placement, wires, nets, coords) {
		// normalize angle --> -180 < angle <= 180
		while (placement.angle <= -180) placement.angle += 360;
		while (placement.angle > 180) placement.angle -= 360;

		// get coordinate
		let instanceCoord = new Coordinate(placement.x * placement.scaling, placement.y * placement.scaling);

		const scaling = {
			x: placement.xScale * placement.scaling,
			y: placement.yScale * placement.scaling,
		};
		const ablTransistorClone = this.clone();
		ablTransistorClone.pins.forEach((pin) => pin.coord.scale(scaling.x, scaling.y));
		ablTransistorClone.rotate(placement.angle);
		ablTransistorClone.mirror(placement.mirrorX, placement.mirrorY);
		ablTransistorClone.translate(instanceCoord);
		// scale

		const transistor = this.tikzTransistor.deepClone();
		transistor.rotate(placement.angle);
		transistor.mirror(placement.mirrorX, placement.mirrorY);
		transistor.translate(ablTransistorClone.lineCrossingCoord.clone().subtract(transistor.lineCrossingCoord));

		if (global.DEBUG) {
			// "Midpoint"
			console.error("\\draw[Gold3] " + transistor.lineCrossingCoord.serializeName() + " circle [radius=1.2pt];");

			// Initial pins of TikZ transistor (SearchHints)
			transistor.pins.forEach((pin) =>
				console.error("\\draw[Cornsilk4] " + pin.coord.serializeName() + " circle [radius=1.2pt];")
			);
		}

		if (transistor.anchorNr != null && transistor.anchorCoord) {
			const oldCoord = coords.find((existingCoord) => transistor.anchorCoord.equals(existingCoord));
			if (oldCoord) transistor.anchorCoord = oldCoord;
			else coords.push(transistor.anchorCoord);
		}

		// set pins (+positions)
		pins.forEach((pin) => pin.findPosition(ablTransistorClone.pins, null, null, wires, nets, coords, false));

		// reorder
		pins = ablTransistorClone.pins.map((ablPin) =>
			pins.find((pin) =>
				pin.name && ablPin.name ? pin.name == ablPin.name : pin.instTermNumber === ablPin.instTermNumber
			)
		);

		// Pins of ADS transistor
		if (global.DEBUG)
			pins.forEach((pin) =>
				console.error("\\draw[SpringGreen4] " + pin.coord.serializeName() + " circle [radius=1.2pt];")
			);

		for (let i = 0; i < transistor.pins.length && i < pins.length; i++) {
			pins[i].coord.x = transistor.pins[i].coord.x;
			pins[i].coord.y = transistor.pins[i].coord.y;
		}

		// Pins of final TikZ Transistor
		if (global.DEBUG)
			pins.forEach((pin) =>
				console.error(
					"\\node at " + pin.coord.serializeName() + " {\\color{Turquoise3}\\pgfuseplotmark{Mercedes star}};"
				)
			);

		transistor.pins = pins;
		transistor.instanceName = instanceName;

		return transistor;
	}
}

export { ABLTransistorMapper };