Fixes and improvements
This commit is contained in:
parent
6d77ff13c9
commit
083a25f302
104
connections.js
104
connections.js
@ -9,16 +9,18 @@ export const connectionStyles = css`
|
||||
transition: stroke 0.3s ease, stroke-width 0.3s ease;
|
||||
pointer-events: all;
|
||||
cursor: pointer;
|
||||
stroke: var(--connection-stroke-color, #999);
|
||||
stroke-width: var(--connection-stroke-width, 3);
|
||||
}
|
||||
|
||||
.connections path:hover {
|
||||
stroke: #999;
|
||||
stroke-width: 3;
|
||||
stroke: var(--connection-stroke-color-hover, #999);
|
||||
stroke-width: var(--connection-stroke-width-hover, 3);
|
||||
}
|
||||
|
||||
.connections path.selected {
|
||||
stroke: #3498db;
|
||||
stroke-width: 3;
|
||||
stroke: var(--connection-stroke-color-selected, #3498db);
|
||||
stroke-width: var(--connection-stroke-width-selected, 3);
|
||||
}
|
||||
|
||||
.delete-button-group {
|
||||
@ -55,45 +57,56 @@ export const connections = function(superClass) {
|
||||
this.connections = [];
|
||||
this.draggingConnection = null;
|
||||
this.mousePosition = { x: 0, y: 0 };
|
||||
this._updatePreviewConnection = this._updatePreviewConnection.bind(this)
|
||||
this._conDocClick = this._conDocClick.bind(this);
|
||||
}
|
||||
|
||||
firstUpdated() {
|
||||
super.firstUpdated();
|
||||
|
||||
this.addEventListener('port-click', this._handlePortClick);
|
||||
document.addEventListener('mousemove', this._updatePreviewConnection.bind(this));
|
||||
document.addEventListener('mousemove', this._updatePreviewConnection);
|
||||
document.addEventListener('click', this._conDocClick);
|
||||
}
|
||||
|
||||
disconnectedCallback() {
|
||||
super.disconnectedCallback();
|
||||
|
||||
document.addEventListener('click', (e) => {
|
||||
// Check if clicking on an output port
|
||||
const path = e.composedPath();
|
||||
const isOutputPort = path.some(el =>
|
||||
this.removeEventListener('port-click', this._handlePortClick);
|
||||
document.removeEventListener('mousemove', this._updatePreviewConnection);
|
||||
document.removeEventListener('click', this._conDocClick);
|
||||
}
|
||||
|
||||
_conDocClick(e) {
|
||||
// Check if clicking on an output port
|
||||
const path = e.composedPath();
|
||||
const isOutputPort = path.some(el =>
|
||||
el instanceof HTMLElement &&
|
||||
el.classList &&
|
||||
el.classList.contains('output-ports')
|
||||
);
|
||||
|
||||
// Only cancel connection if we're dragging AND it's not an output port
|
||||
if (this.draggingConnection && !isOutputPort) {
|
||||
// Check if we clicked on an input port
|
||||
const isInputPort = path.some(el =>
|
||||
el instanceof HTMLElement &&
|
||||
el.classList &&
|
||||
el.classList.contains('output-ports')
|
||||
el.classList.contains('input-ports')
|
||||
);
|
||||
|
||||
// Only cancel connection if we're dragging AND it's not an output port
|
||||
if (this.draggingConnection && !isOutputPort) {
|
||||
// Check if we clicked on an input port
|
||||
const isInputPort = path.some(el =>
|
||||
el instanceof HTMLElement &&
|
||||
el.classList &&
|
||||
el.classList.contains('input-ports')
|
||||
);
|
||||
|
||||
// If not an input port, cancel the connection
|
||||
if (!isInputPort) {
|
||||
this.draggingConnection = null;
|
||||
this.requestUpdate();
|
||||
}
|
||||
}
|
||||
|
||||
// Handle existing connection deselection
|
||||
if (e.target === this.canvas) {
|
||||
this.selectedConnection = null;
|
||||
// If not an input port, cancel the connection
|
||||
if (!isInputPort) {
|
||||
this.draggingConnection = null;
|
||||
this.requestUpdate();
|
||||
}
|
||||
});
|
||||
|
||||
super.firstUpdated();
|
||||
}
|
||||
|
||||
// Handle existing connection deselection
|
||||
if (e.target === this.canvas) {
|
||||
this.selectedConnection = null;
|
||||
this.requestUpdate();
|
||||
}
|
||||
}
|
||||
|
||||
_updatePreviewConnection(e) {
|
||||
@ -117,6 +130,7 @@ export const connections = function(superClass) {
|
||||
width: ${bounds.width}px;
|
||||
height: ${bounds.height}px;
|
||||
pointer-events: none;
|
||||
z-index: -1;
|
||||
` : `
|
||||
position: absolute;
|
||||
top: 0;
|
||||
@ -124,10 +138,11 @@ export const connections = function(superClass) {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
pointer-events: none;
|
||||
z-index: -1;
|
||||
`;
|
||||
|
||||
return html`
|
||||
<svg
|
||||
<svg
|
||||
class="connections"
|
||||
style="${style}"
|
||||
viewBox="${this.canvas ? `${bounds.minX} ${bounds.minY} ${bounds.width} ${bounds.height}` : '0 0 100 100'}"
|
||||
@ -143,7 +158,7 @@ export const connections = function(superClass) {
|
||||
if (!sourceNode) return '';
|
||||
|
||||
const sourcePort = sourceNode.shadowRoot.querySelector(
|
||||
`.output-ports [data-port-id="${this.draggingConnection.sourcePortId}"]`
|
||||
`.output-ports [portid="${this.draggingConnection.sourcePortId}"]`
|
||||
);
|
||||
if (!sourcePort) return '';
|
||||
|
||||
@ -190,8 +205,8 @@ export const connections = function(superClass) {
|
||||
|
||||
if (!sourceNode?.shadowRoot || !targetNode?.shadowRoot) return '';
|
||||
|
||||
const sourcePort = sourceNode.shadowRoot.querySelector(`.output-ports [data-port-id="${conn.sourcePortId}"]`);
|
||||
const targetPort = targetNode.shadowRoot.querySelector(`.input-ports [data-port-id="${conn.targetPortId}"]`);
|
||||
const sourcePort = sourceNode.shadowRoot.querySelector(`.output-ports [portid="${conn.sourcePortId}"]`);
|
||||
const targetPort = targetNode.shadowRoot.querySelector(`.input-ports [portid="${conn.targetPortId}"]`);
|
||||
|
||||
if (!sourcePort || !targetPort) return '';
|
||||
|
||||
@ -312,18 +327,31 @@ export const connections = function(superClass) {
|
||||
}
|
||||
|
||||
_handlePortClick(e) {
|
||||
const { nodeId, portType, portId, portName } = e.detail;
|
||||
const { node, port } = e.detail;
|
||||
const nodeId = node.id;
|
||||
const { portType, portId, portName } = port;
|
||||
|
||||
if (!this.draggingConnection) {
|
||||
if (portType === 'output') {
|
||||
this.draggingConnection = {
|
||||
sourceNodeId: nodeId,
|
||||
sourcePortId: portId,
|
||||
sourcePortType: portType
|
||||
sourcePortType: portType,
|
||||
sourceNode: node
|
||||
};
|
||||
}
|
||||
} else {
|
||||
if (portType === 'input' && nodeId !== this.draggingConnection.sourceNodeId) {
|
||||
const targetNode = node;
|
||||
const sourceNode = this.draggingConnection.sourceNode;
|
||||
|
||||
// Validate the connection
|
||||
const isValid = targetNode.validateConnection(sourceNode, this.draggingConnection.sourcePortId, portId);
|
||||
|
||||
if (!isValid) {
|
||||
return;
|
||||
}
|
||||
|
||||
const connectionExists = this.connections.some(conn =>
|
||||
conn.sourceNodeId === this.draggingConnection.sourceNodeId &&
|
||||
conn.sourcePortId === this.draggingConnection.sourcePortId &&
|
||||
|
@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@tp/tp-flow-nodes",
|
||||
"version": "1.0.0",
|
||||
"version": "1.1.0",
|
||||
"description": "",
|
||||
"main": "tp-flow-nodes.js",
|
||||
"scripts": {
|
||||
@ -13,6 +13,8 @@
|
||||
"author": "trading_peter",
|
||||
"license": "Apache-2.0",
|
||||
"dependencies": {
|
||||
"lit": "^3.0.0"
|
||||
"lit": "^3.0.0",
|
||||
"@tp/helpers": "^2.0.0",
|
||||
"@tp/tp-timeout-strip": "^1.0.0"
|
||||
}
|
||||
}
|
||||
|
32
panning.js
32
panning.js
@ -7,6 +7,8 @@ export const panning = function(superClass) {
|
||||
isDragging: { type: Boolean },
|
||||
currentX: { type: Number },
|
||||
currentY: { type: Number },
|
||||
currentNodeX: { type: Number },
|
||||
currentNodeY: { type: Number },
|
||||
initialX: { type: Number },
|
||||
initialY: { type: Number },
|
||||
xOffset: { type: Number },
|
||||
@ -20,6 +22,8 @@ export const panning = function(superClass) {
|
||||
this.isDragging = false;
|
||||
this.currentX = 0;
|
||||
this.currentY = 0;
|
||||
this.currentNodeX = 0;
|
||||
this.currentNodeY = 0;
|
||||
this.initialX = 0;
|
||||
this.initialY = 0;
|
||||
this.xOffset = 0;
|
||||
@ -51,8 +55,17 @@ export const panning = function(superClass) {
|
||||
_bringToFront(element) {
|
||||
if (!(element instanceof TpFlowNode)) return;
|
||||
|
||||
this.highestZIndex++;
|
||||
element.style.zIndex = this.highestZIndex;
|
||||
// Get all elements and filter for TpFlowNode instances
|
||||
const nodes = Array.from(this.shadowRoot.querySelector('.canvas').children)
|
||||
.filter(node => node instanceof TpFlowNode);
|
||||
|
||||
for (const node of nodes) {
|
||||
node.style.zIndex = 1;
|
||||
node.classList.remove('focused');
|
||||
}
|
||||
|
||||
element.style.zIndex = 2;
|
||||
element.classList.add('focused');
|
||||
}
|
||||
|
||||
_startDrag(e) {
|
||||
@ -61,6 +74,11 @@ export const panning = function(superClass) {
|
||||
if (topNode) {
|
||||
this.targetElement = topNode;
|
||||
this._bringToFront(topNode);
|
||||
|
||||
// Check if a element with the "drag-node" attribute is part of the event path. Only then we can start dragging.
|
||||
if (!e.composedPath().some(el => typeof el.hasAttribute === 'function' && el.hasAttribute('drag-node'))) {
|
||||
return;
|
||||
}
|
||||
} else {
|
||||
this.targetElement = this.canvas;
|
||||
}
|
||||
@ -96,10 +114,10 @@ export const panning = function(superClass) {
|
||||
`translate(${this.currentX}px, ${this.currentY}px) scale(${this.scale})`;
|
||||
} else {
|
||||
// For nodes, compensate for canvas scale
|
||||
this.currentX = (e.clientX - this.initialX) / this.scale;
|
||||
this.currentY = (e.clientY - this.initialY) / this.scale;
|
||||
this.currentNodeX = (e.clientX - this.initialX) / this.scale;
|
||||
this.currentNodeY = (e.clientY - this.initialY) / this.scale;
|
||||
this.targetElement.style.transform =
|
||||
`translate(${this.currentX}px, ${this.currentY}px)`;
|
||||
`translate(${this.currentNodeX}px, ${this.currentNodeY}px)`;
|
||||
}
|
||||
|
||||
this.requestUpdate();
|
||||
@ -113,8 +131,8 @@ export const panning = function(superClass) {
|
||||
data: {
|
||||
nodeId: this.targetElement.id,
|
||||
position: {
|
||||
x: this.currentX,
|
||||
y: this.currentY
|
||||
x: this.currentNodeX,
|
||||
y: this.currentNodeY
|
||||
}
|
||||
}
|
||||
});
|
||||
|
109
tp-flow-node-port.js
Normal file
109
tp-flow-node-port.js
Normal file
@ -0,0 +1,109 @@
|
||||
/**
|
||||
@license
|
||||
Copyright (c) 2025 trading_peter
|
||||
This program is available under Apache License Version 2.0
|
||||
*/
|
||||
|
||||
import '@tp/tp-timeout-strip/tp-timeout-strip.js';
|
||||
import { LitElement, html, css } from 'lit';
|
||||
|
||||
class TpFlowNodePort extends LitElement {
|
||||
static get styles() {
|
||||
return [
|
||||
css`
|
||||
:host {
|
||||
display: block;
|
||||
position: relative;
|
||||
}
|
||||
|
||||
.connectionPoint {
|
||||
width: 12px;
|
||||
height: 12px;
|
||||
background: #666;
|
||||
border-radius: 50%;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.connectionPoint:hover {
|
||||
background: #888;
|
||||
}
|
||||
|
||||
.tag {
|
||||
position: absolute;
|
||||
background: #888;
|
||||
visibility: hidden;
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
:host([portType="input"]) .tag {
|
||||
left: 0;
|
||||
top: 0;
|
||||
transform: translateX(calc(-100% - 10px));
|
||||
}
|
||||
|
||||
:host([portType="output"]) .tag {
|
||||
right: 0;
|
||||
top: 0;
|
||||
transform: translateX(calc(100% + 10px));
|
||||
}
|
||||
|
||||
:host(:hover) .tag,
|
||||
.tag[visible] {
|
||||
visibility: visible;
|
||||
pointer-events: all;
|
||||
}
|
||||
`
|
||||
];
|
||||
}
|
||||
|
||||
render() {
|
||||
const { showTag, errorMsg, tagContent } = this;
|
||||
|
||||
return html`
|
||||
<div class="connectionPoint" part="connection-point"></div>
|
||||
|
||||
${showTag || Boolean(tagContent) ? html`
|
||||
<div class="tag" ?visible=${showTag} part="tag">
|
||||
${errorMsg ? html`
|
||||
<div part="tag-error">
|
||||
${errorMsg}
|
||||
</div>
|
||||
` : null}
|
||||
${tagContent ? html`
|
||||
<div class="tag-content" part="tag-content">${tagContent}</div>
|
||||
` : null}
|
||||
<tp-timeout-strip part="tag-timeout" @timeout=${this._onTimeout}></tp-timeout-strip>
|
||||
</div>
|
||||
` : null}
|
||||
`;
|
||||
}
|
||||
|
||||
static get properties() {
|
||||
return {
|
||||
portType: { type: String, reflect: true },
|
||||
portId: { type: Number, reflect: true },
|
||||
portName: { type: String },
|
||||
tagContent: { type: String },
|
||||
errorMsg: { type: String },
|
||||
showTag: { type: Boolean },
|
||||
};
|
||||
}
|
||||
|
||||
showConnectionError(msg, timeout = 0) {
|
||||
this.errorMsg = msg;
|
||||
this.showTag = true;
|
||||
|
||||
if (timeout > 0) {
|
||||
this.updateComplete.then(() => {
|
||||
this.shadowRoot.querySelector('tp-timeout-strip').show(timeout);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
_onTimeout() {
|
||||
this.showTag = false;
|
||||
this.errorMsg = null;
|
||||
}
|
||||
}
|
||||
|
||||
window.customElements.define('tp-flow-node-port', TpFlowNodePort);
|
@ -4,6 +4,7 @@ Copyright (c) 2024 trading_peter
|
||||
This program is available under Apache License Version 2.0
|
||||
*/
|
||||
|
||||
import './tp-flow-node-port.js';
|
||||
import { LitElement, html, css } from 'lit';
|
||||
|
||||
// tp-flow-node.js
|
||||
@ -46,24 +47,14 @@ export class TpFlowNode extends LitElement {
|
||||
align-items: flex-end;
|
||||
}
|
||||
|
||||
.port {
|
||||
width: 12px;
|
||||
height: 12px;
|
||||
background: #666;
|
||||
border-radius: 50%;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.port:hover {
|
||||
background: #888;
|
||||
header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
column-gap: 10px;
|
||||
padding: 2px 5px;
|
||||
}
|
||||
|
||||
.delete-btn {
|
||||
position: absolute;
|
||||
top: 5px;
|
||||
right: 5px;
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
cursor: pointer;
|
||||
opacity: 0.7;
|
||||
transition: opacity 0.2s;
|
||||
@ -78,16 +69,17 @@ export class TpFlowNode extends LitElement {
|
||||
|
||||
render() {
|
||||
return html`
|
||||
${this.renderNodeHeader()}
|
||||
<div class="node-body">
|
||||
<div class="delete-btn" @click="${this._handleDelete}">✕</div>
|
||||
<div class="input-ports">
|
||||
${this.inputs.map((input, idx) => html`
|
||||
<div class="port"
|
||||
data-port-type="input"
|
||||
data-port-id="${idx}"
|
||||
data-port-name="${input.name}"
|
||||
@mousedown="${this._handlePortClick}">
|
||||
</div>
|
||||
<tp-flow-node-port class="port" exportparts="connectionPoint"
|
||||
portType="input"
|
||||
.portId=${idx}
|
||||
.portName=${input.name}
|
||||
.tagContent=${input.tagContent}
|
||||
@mousedown="${this._handlePortClick}">
|
||||
</tp-flow-node-port>
|
||||
`)}
|
||||
</div>
|
||||
|
||||
@ -97,18 +89,28 @@ export class TpFlowNode extends LitElement {
|
||||
|
||||
<div class="output-ports">
|
||||
${this.outputs.map((output, idx) => html`
|
||||
<div class="port"
|
||||
data-port-type="output"
|
||||
data-port-id="${idx}"
|
||||
data-port-name="${output.name}"
|
||||
@mousedown="${this._handlePortClick}">
|
||||
</div>
|
||||
<tp-flow-node-port class="port" exportparts="connectionPoint"
|
||||
portType="output"
|
||||
.portId=${idx}
|
||||
.portName=${output.name}
|
||||
.tagContent=${output.tagContent}
|
||||
@mousedown="${this._handlePortClick}">
|
||||
</tp-flow-node-port>
|
||||
`)}
|
||||
</div>
|
||||
</div>
|
||||
`;
|
||||
}
|
||||
|
||||
renderNodeHeader() {
|
||||
return html`
|
||||
<header drag-node>
|
||||
A Node
|
||||
<div class="delete-btn" @click="${this._handleDelete}">✕</div>
|
||||
</header>
|
||||
`;
|
||||
}
|
||||
|
||||
renderNodeContent() {
|
||||
console.warn('Your node should override the renderNodeContent method.');
|
||||
return null;
|
||||
@ -133,19 +135,17 @@ export class TpFlowNode extends LitElement {
|
||||
this.data = {};
|
||||
}
|
||||
|
||||
updated(changes) {
|
||||
super.updated(changes);
|
||||
|
||||
this.dispatchEvent(new CustomEvent('update-layout', { detail: this, bubbles: true, composed: true }));
|
||||
}
|
||||
|
||||
_handlePortClick(e) {
|
||||
e.stopPropagation(); // Prevents the event from getting caught by the panning action.
|
||||
|
||||
const portEl = e.target;
|
||||
const detail = {
|
||||
nodeId: this.id,
|
||||
portType: portEl.dataset.portType,
|
||||
portId: portEl.dataset.portId,
|
||||
portName: portEl.dataset.portName
|
||||
};
|
||||
|
||||
this.dispatchEvent(new CustomEvent('port-click', {
|
||||
detail,
|
||||
detail: { node: this, port: e.target },
|
||||
bubbles: true,
|
||||
composed: true
|
||||
}));
|
||||
|
@ -8,6 +8,7 @@ import { LitElement, html, css } from 'lit';
|
||||
import { connections, connectionStyles } from './connections.js';
|
||||
import { panning } from './panning.js';
|
||||
import { zoom } from './zoom.js';
|
||||
import { repeat } from 'lit/directives/repeat.js';
|
||||
|
||||
export class TpFlowNodes extends zoom(panning(connections(LitElement))) {
|
||||
static nodeTypes = new Map();
|
||||
@ -36,10 +37,11 @@ export class TpFlowNodes extends zoom(panning(connections(LitElement))) {
|
||||
];
|
||||
}
|
||||
|
||||
// Using the repeat directive to render the nodes is important here to prevent the nodes from running into a inconsistent data state when nodes are removed.
|
||||
render() {
|
||||
return html`
|
||||
<div class="canvas">
|
||||
${this.nodes.map(node => html`${node}`)}
|
||||
${repeat(this.nodes, node => node.id, node => html`${node}`)}
|
||||
${this._renderConnections()}
|
||||
</div>
|
||||
`;
|
||||
@ -61,6 +63,9 @@ export class TpFlowNodes extends zoom(panning(connections(LitElement))) {
|
||||
connectedCallback() {
|
||||
super.connectedCallback();
|
||||
this.addEventListener('node-delete-requested', this._boundDeleteHandler);
|
||||
this.addEventListener('update-layout', () => {
|
||||
this.requestUpdate();
|
||||
})
|
||||
}
|
||||
|
||||
disconnectedCallback() {
|
||||
|
Loading…
x
Reference in New Issue
Block a user