|
|
|
class Cable {
|
|
|
|
start = "";
|
|
|
|
end = "";
|
|
|
|
strPath = "";
|
|
|
|
color = "";
|
|
|
|
strokeWidth = 10;
|
|
|
|
bufferSize = 20;
|
|
|
|
buffer = [];
|
|
|
|
path = null;
|
|
|
|
|
|
|
|
constructor(start) {
|
|
|
|
this.start = start;
|
|
|
|
this.randomColor();
|
|
|
|
}
|
|
|
|
|
|
|
|
randomColor() {
|
|
|
|
const hue = Math.floor(Math.random() * 360);
|
|
|
|
const saturation = Math.floor(50 + Math.random() * (50 + 1)) + "%";
|
|
|
|
const lightness = "75%";
|
|
|
|
this.color = "hsl(" + hue + ", " + saturation + ", " + lightness + ")";
|
|
|
|
}
|
|
|
|
|
|
|
|
createCable(position) {
|
|
|
|
this.path = document.createElementNS("http://www.w3.org/2000/svg", "path");
|
|
|
|
|
|
|
|
this.path.setAttribute("fill", "none");
|
|
|
|
|
|
|
|
this.path.setAttribute("stroke", this.color);
|
|
|
|
this.path.setAttribute("stroke-width", this.strokeWidth);
|
|
|
|
this.path.setAttribute("stroke-linecap", "round");
|
|
|
|
|
|
|
|
this.buffer = [];
|
|
|
|
this.appendToBuffer(position);
|
|
|
|
this.strPath = "M" + position.x + " " + position.y;
|
|
|
|
this.path.setAttribute("d", this.strPath);
|
|
|
|
}
|
|
|
|
|
|
|
|
appendToBuffer(position) {
|
|
|
|
this.buffer.push(position);
|
|
|
|
while (this.buffer.length > this.bufferSize) {
|
|
|
|
this.buffer.shift();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getAveragePoint(offset) {
|
|
|
|
// Calculate the average point, starting at offset in the buffer
|
|
|
|
let len = this.buffer.length;
|
|
|
|
if (len % 2 === 1 || len >= this.bufferSize) {
|
|
|
|
let totalX = 0;
|
|
|
|
let totalY = 0;
|
|
|
|
let position, i;
|
|
|
|
let count = 0;
|
|
|
|
for (i = offset; i < len; i++) {
|
|
|
|
count++;
|
|
|
|
position = this.buffer[i];
|
|
|
|
totalX += position.x;
|
|
|
|
totalY += position.y;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
x: totalX / count,
|
|
|
|
y: totalY / count,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
updatePath() {
|
|
|
|
let position = this.getAveragePoint(0);
|
|
|
|
if (position) {
|
|
|
|
// Get the smoothed part of the path that will not change
|
|
|
|
this.strPath += " L" + position.x + " " + position.y;
|
|
|
|
|
|
|
|
// Get the last part of the path (close to the current mouse position)
|
|
|
|
// This part will change if the mouse moves again
|
|
|
|
var tmpPath = "";
|
|
|
|
for (var offset = 2; offset < this.buffer.length; offset += 2) {
|
|
|
|
position = this.getAveragePoint(offset);
|
|
|
|
tmpPath += " L" + position.x + " " + position.y;
|
|
|
|
}
|
|
|
|
this.path.setAttribute("d", this.strPath + tmpPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Panel {
|
|
|
|
width = 0;
|
|
|
|
height = 0;
|
|
|
|
|
|
|
|
svg = null;
|
|
|
|
container = null;
|
|
|
|
containerBoundingClient = null;
|
|
|
|
|
|
|
|
model = {};
|
|
|
|
params = [];
|
|
|
|
sockets = [];
|
|
|
|
preset = {};
|
|
|
|
|
|
|
|
cable = null;
|
|
|
|
cables = null;
|
|
|
|
|
|
|
|
constructor(svg, container, preset = {}, debug = false) {
|
|
|
|
this.svg = this.htmlToElement(svg);
|
|
|
|
this.container = container;
|
|
|
|
this.container.style.position = "relative";
|
|
|
|
this.containerRect = container.getBoundingClientRect();
|
|
|
|
this.preset = { ...preset };
|
|
|
|
this.debug = debug;
|
|
|
|
|
|
|
|
while (container.firstChild) {
|
|
|
|
container.removeChild(container.lastChild);
|
|
|
|
}
|
|
|
|
container.appendChild(this.svg);
|
|
|
|
|
|
|
|
this.setSize();
|
|
|
|
this.createParams();
|
|
|
|
|
|
|
|
this.createSockets();
|
|
|
|
this.createCables();
|
|
|
|
|
|
|
|
this.svg.addEventListener("mousedown", (e) => this.startCable(e));
|
|
|
|
this.svg.addEventListener("mousemove", (e) => this.drawCable(e));
|
|
|
|
this.svg.addEventListener("mouseup", (e) => this.endCable(e));
|
|
|
|
// this.svg.addEventListener("mouseleave", (e) => this.endCable(e));
|
|
|
|
}
|
|
|
|
|
|
|
|
htmlToElement(string) {
|
|
|
|
var template = document.createElement("template");
|
|
|
|
string = string.trim(); // Never return a text node of whitespace as the result
|
|
|
|
template.innerHTML = string;
|
|
|
|
return template.content.firstChild;
|
|
|
|
}
|
|
|
|
|
|
|
|
setSize() {
|
|
|
|
this.width = this.svg.getAttribute("width");
|
|
|
|
this.height = this.svg.getAttribute("height");
|
|
|
|
this.container.style.width = this.width + "px";
|
|
|
|
this.container.style.height = this.height + "px";
|
|
|
|
}
|
|
|
|
|
|
|
|
createParams() {
|
|
|
|
// this.svg.querySelector("#params").style.visibility = "hidden";
|
|
|
|
this.params = this.svg.querySelectorAll("#params [fill='#FF0000']");
|
|
|
|
this.model.params = [];
|
|
|
|
|
|
|
|
for (const param of this.params) {
|
|
|
|
let rect = param.getBoundingClientRect();
|
|
|
|
|
|
|
|
let control = document.createElement("input");
|
|
|
|
control.setAttribute("type", "range");
|
|
|
|
control.setAttribute("data-width", rect.width);
|
|
|
|
control.setAttribute("data-height", rect.height);
|
|
|
|
|
|
|
|
control.setAttribute("data-fgcolor", "white");
|
|
|
|
control.classList.add("input-knob");
|
|
|
|
control.style.position = "absolute";
|
|
|
|
control.style.left = rect.left - this.containerRect.left + "px";
|
|
|
|
control.style.top = rect.top - this.containerRect.top + "px";
|
|
|
|
|
|
|
|
control.setAttribute("name", param.id);
|
|
|
|
|
|
|
|
if (this.preset.hasOwnProperty(param.id)) {
|
|
|
|
control.value = this.preset[param.id];
|
|
|
|
}
|
|
|
|
|
|
|
|
this.container.appendChild(control);
|
|
|
|
|
|
|
|
this.model.params.push(param.id);
|
|
|
|
|
|
|
|
if (this.debug) {
|
|
|
|
let label = document.createElement("label");
|
|
|
|
label.setAttribute("for", param.id);
|
|
|
|
label.innerHTML = param.id;
|
|
|
|
|
|
|
|
label.style.position = "absolute";
|
|
|
|
label.style.left = rect.left - this.containerRect.left + "px";
|
|
|
|
label.style.top = rect.top + rect.height + "px";
|
|
|
|
label.style.fontSize = "1rem";
|
|
|
|
label.style.backgroundColor = "red";
|
|
|
|
label.style.color = "white";
|
|
|
|
|
|
|
|
this.container.appendChild(label);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let group = this.svg.querySelector("#params");
|
|
|
|
if (group) group.style.display = "none";
|
|
|
|
}
|
|
|
|
|
|
|
|
createSockets() {
|
|
|
|
// this.svg.querySelector("#sockets").style.visibility = "hidden";
|
|
|
|
this.sockets = this.svg.querySelectorAll("#sockets [fill='#00FF00']");
|
|
|
|
this.model.sockets = [];
|
|
|
|
for (const socket of this.sockets) {
|
|
|
|
let rect = socket.getBoundingClientRect();
|
|
|
|
|
|
|
|
let input = document.createElement("input");
|
|
|
|
input.setAttribute("name", socket.id);
|
|
|
|
input.classList.add("socket");
|
|
|
|
|
|
|
|
input.style.position = "absolute";
|
|
|
|
input.style.left = rect.left - this.containerRect.left + "px";
|
|
|
|
input.style.top = rect.top - this.containerRect.top + "px";
|
|
|
|
input.style.width = rect.width + "px";
|
|
|
|
input.style.height = rect.height + "px";
|
|
|
|
|
|
|
|
input.style.border = "none";
|
|
|
|
input.style.background = "none";
|
|
|
|
input.style.opacity = 0;
|
|
|
|
|
|
|
|
if (this.debug) {
|
|
|
|
input.style.border = "1px solid #00FF00";
|
|
|
|
input.style.opacity = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
input.style.cursor = "alias";
|
|
|
|
|
|
|
|
input.addEventListener("mouseup", (e) => this.endCable(e));
|
|
|
|
|
|
|
|
input.addEventListener("mousedown", (e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
this.startCable(e);
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
|
|
|
|
this.container.appendChild(input);
|
|
|
|
|
|
|
|
this.model.sockets.push(socket.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
let group = this.svg.querySelector("#sockets");
|
|
|
|
if (group) group.style.display = "none";
|
|
|
|
}
|
|
|
|
|
|
|
|
createCables() {
|
|
|
|
let cables = document.createElementNS("http://www.w3.org/2000/svg", "svg");
|
|
|
|
|
|
|
|
cables.classList.add("cables");
|
|
|
|
cables.setAttribute("xmlns", "http://www.w3.org/2000/svg");
|
|
|
|
cables.setAttribute("width", this.width);
|
|
|
|
cables.setAttribute("height", this.height);
|
|
|
|
cables.setAttribute("viewBox", `0 0 ${this.width} ${this.height}`);
|
|
|
|
cables.setAttribute("fill", "none");
|
|
|
|
cables.style.pointerEvents = "none";
|
|
|
|
cables.style.position = "absolute";
|
|
|
|
cables.style.left = 0;
|
|
|
|
cables.style.top = 0;
|
|
|
|
cables.style.width = this.width;
|
|
|
|
cables.style.height = this.height;
|
|
|
|
|
|
|
|
this.cables = cables;
|
|
|
|
this.container.appendChild(this.cables);
|
|
|
|
}
|
|
|
|
|
|
|
|
startCable(event) {
|
|
|
|
let position = this.getMousePosition(event);
|
|
|
|
let socket = event.target;
|
|
|
|
|
|
|
|
if (socket && socket.classList.contains("socket")) {
|
|
|
|
this.cable = new Cable(socket.getAttribute("name"));
|
|
|
|
this.cable.createCable(position);
|
|
|
|
let cable = this.cable.path;
|
|
|
|
cable.style.pointerEvents = "stroke";
|
|
|
|
|
|
|
|
cable.addEventListener("click", (e) => {
|
|
|
|
console.log(e);
|
|
|
|
cable.remove();
|
|
|
|
});
|
|
|
|
|
|
|
|
this.cables.appendChild(this.cable.path);
|
|
|
|
|
|
|
|
for (const input of this.container.querySelectorAll(".input-knob")) {
|
|
|
|
input.style.pointerEvents = "none";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
drawCable(event) {
|
|
|
|
if (this.cable) {
|
|
|
|
this.cable.appendToBuffer(this.getMousePosition(event));
|
|
|
|
this.cable.updatePath();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
endCable(event) {
|
|
|
|
if (this.cable) {
|
|
|
|
let socket = event.target;
|
|
|
|
|
|
|
|
if (socket && socket.classList.contains("socket")) {
|
|
|
|
this.cable.end = socket.getAttribute("name");
|
|
|
|
|
|
|
|
let startSocket = document.querySelector(`.socket[name="${this.cable.start}"]`);
|
|
|
|
|
|
|
|
let values = (startSocket.getAttribute("value") || "")
|
|
|
|
.split(",")
|
|
|
|
.map((entry) => entry.trim());
|
|
|
|
values.push(this.cable.end);
|
|
|
|
startSocket.setAttribute("value", values.filter((value) => value != "").join());
|
|
|
|
|
|
|
|
// TODO: log cable
|
|
|
|
} else {
|
|
|
|
this.cable.path.remove();
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const input of this.container.querySelectorAll(".input-knob")) {
|
|
|
|
input.style.pointerEvents = "all";
|
|
|
|
}
|
|
|
|
|
|
|
|
this.cable = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getMousePosition(e) {
|
|
|
|
return {
|
|
|
|
x: e.pageX - this.containerRect.left,
|
|
|
|
y: e.pageY - this.containerRect.top,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|