wip
This commit is contained in:
parent
6af27de3dd
commit
a892e1fcb2
@ -101,13 +101,13 @@ export const connections = function(superClass) {
|
|||||||
const canvasRect = this.canvas.getBoundingClientRect();
|
const canvasRect = this.canvas.getBoundingClientRect();
|
||||||
|
|
||||||
const start = {
|
const start = {
|
||||||
x: sourceRect.left + sourceRect.width/2 - canvasRect.left,
|
x: (sourceRect.left + sourceRect.width/2 - canvasRect.left) / this.scale,
|
||||||
y: sourceRect.top + sourceRect.height/2 - canvasRect.top
|
y: (sourceRect.top + sourceRect.height/2 - canvasRect.top) / this.scale
|
||||||
};
|
};
|
||||||
|
|
||||||
const end = {
|
const end = {
|
||||||
x: this.mousePosition.x - canvasRect.left,
|
x: (this.mousePosition.x - canvasRect.left) / this.scale,
|
||||||
y: this.mousePosition.y - canvasRect.top
|
y: (this.mousePosition.y - canvasRect.top) / this.scale
|
||||||
};
|
};
|
||||||
|
|
||||||
const controlPoint1 = {
|
const controlPoint1 = {
|
||||||
@ -150,13 +150,13 @@ export const connections = function(superClass) {
|
|||||||
const canvasRect = this.canvas.getBoundingClientRect();
|
const canvasRect = this.canvas.getBoundingClientRect();
|
||||||
|
|
||||||
const start = {
|
const start = {
|
||||||
x: sourceRect.left + sourceRect.width/2 - canvasRect.left,
|
x: (sourceRect.left + sourceRect.width/2 - canvasRect.left) / this.scale,
|
||||||
y: sourceRect.top + sourceRect.height/2 - canvasRect.top
|
y: (sourceRect.top + sourceRect.height/2 - canvasRect.top) / this.scale
|
||||||
};
|
};
|
||||||
|
|
||||||
const end = {
|
const end = {
|
||||||
x: targetRect.left + targetRect.width/2 - canvasRect.left,
|
x: (targetRect.left + targetRect.width/2 - canvasRect.left) / this.scale,
|
||||||
y: targetRect.top + targetRect.height/2 - canvasRect.top
|
y: (targetRect.top + targetRect.height/2 - canvasRect.top) / this.scale
|
||||||
};
|
};
|
||||||
|
|
||||||
const controlPoint1 = {
|
const controlPoint1 = {
|
||||||
|
16
panning.js
16
panning.js
@ -73,8 +73,14 @@ export const panning = function(superClass) {
|
|||||||
this.xOffset = matrix.m41;
|
this.xOffset = matrix.m41;
|
||||||
this.yOffset = matrix.m42;
|
this.yOffset = matrix.m42;
|
||||||
|
|
||||||
|
if (this.targetElement === this.canvas) {
|
||||||
this.initialX = e.clientX - this.xOffset;
|
this.initialX = e.clientX - this.xOffset;
|
||||||
this.initialY = e.clientY - this.yOffset;
|
this.initialY = e.clientY - this.yOffset;
|
||||||
|
} else {
|
||||||
|
// For nodes, compensate for scale in initial position too
|
||||||
|
this.initialX = e.clientX - (this.xOffset * this.scale);
|
||||||
|
this.initialY = e.clientY - (this.yOffset * this.scale);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -82,13 +88,21 @@ export const panning = function(superClass) {
|
|||||||
if (this.isDragging && this.targetElement) {
|
if (this.isDragging && this.targetElement) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
|
if (this.targetElement === this.canvas) {
|
||||||
|
// For canvas panning, no scale compensation needed
|
||||||
this.currentX = e.clientX - this.initialX;
|
this.currentX = e.clientX - this.initialX;
|
||||||
this.currentY = e.clientY - this.initialY;
|
this.currentY = e.clientY - this.initialY;
|
||||||
|
this.targetElement.style.transform =
|
||||||
|
`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.targetElement.style.transform =
|
this.targetElement.style.transform =
|
||||||
`translate(${this.currentX}px, ${this.currentY}px)`;
|
`translate(${this.currentX}px, ${this.currentY}px)`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_endDrag() {
|
_endDrag() {
|
||||||
this.isDragging = false;
|
this.isDragging = false;
|
||||||
|
@ -7,8 +7,9 @@ This program is available under Apache License Version 2.0
|
|||||||
import { LitElement, html, css } from 'lit';
|
import { LitElement, html, css } from 'lit';
|
||||||
import { connections, connectionStyles } from './connections.js';
|
import { connections, connectionStyles } from './connections.js';
|
||||||
import { panning } from './panning.js';
|
import { panning } from './panning.js';
|
||||||
|
import { zoom } from './zoom.js';
|
||||||
|
|
||||||
class TpFlowNodes extends panning(connections(LitElement)) {
|
class TpFlowNodes extends zoom(panning(connections(LitElement))) {
|
||||||
static get styles() {
|
static get styles() {
|
||||||
return [
|
return [
|
||||||
connectionStyles,
|
connectionStyles,
|
||||||
@ -23,8 +24,10 @@ class TpFlowNodes extends panning(connections(LitElement)) {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
position: relative;
|
position: absolute;
|
||||||
overflow: hidden;
|
min-width: 100%;
|
||||||
|
min-height: 100%;
|
||||||
|
overflow: visible;
|
||||||
transform: translate(0px, 0px);
|
transform: translate(0px, 0px);
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
91
zoom.js
Normal file
91
zoom.js
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
export const zoom = function(superClass) {
|
||||||
|
return class extends superClass {
|
||||||
|
static get properties() {
|
||||||
|
return {
|
||||||
|
scale: { type: Number }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
this.scale = 1;
|
||||||
|
this._handleWheel = this._handleWheel.bind(this);
|
||||||
|
this.MIN_SCALE = 0.1;
|
||||||
|
this.MAX_SCALE = 1.0;
|
||||||
|
this.SCALE_STEP = 0.1;
|
||||||
|
}
|
||||||
|
|
||||||
|
firstUpdated() {
|
||||||
|
super.firstUpdated();
|
||||||
|
this.canvas = this.shadowRoot.querySelector('.canvas');
|
||||||
|
this.addEventListener('wheel', this._handleWheel, { passive: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
_handleWheel(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
if (e.deltaY > 0) {
|
||||||
|
this.zoomOut();
|
||||||
|
} else if (e.deltaY < 0) {
|
||||||
|
this.zoomIn();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Zooms out the canvas by one step
|
||||||
|
* @returns {boolean} True if zoom was applied, false if already at minimum
|
||||||
|
*/
|
||||||
|
zoomOut() {
|
||||||
|
const newScale = Math.max(this.MIN_SCALE, this.scale - this.SCALE_STEP);
|
||||||
|
if (newScale !== this.scale) {
|
||||||
|
this._applyZoom(newScale);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Zooms in the canvas by one step
|
||||||
|
* @returns {boolean} True if zoom was applied, false if already at maximum
|
||||||
|
*/
|
||||||
|
zoomIn() {
|
||||||
|
const newScale = Math.min(this.MAX_SCALE, this.scale + this.SCALE_STEP);
|
||||||
|
if (newScale !== this.scale) {
|
||||||
|
this._applyZoom(newScale);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Immediately resets zoom to 100%
|
||||||
|
*/
|
||||||
|
resetZoom() {
|
||||||
|
this._applyZoom(this.MAX_SCALE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Applies the zoom scale to the canvas
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
_applyZoom(newScale) {
|
||||||
|
this.scale = newScale;
|
||||||
|
|
||||||
|
// Get current translation
|
||||||
|
const transform = window.getComputedStyle(this.canvas).transform;
|
||||||
|
const matrix = new DOMMatrix(transform);
|
||||||
|
const currentX = matrix.m41;
|
||||||
|
const currentY = matrix.m42;
|
||||||
|
|
||||||
|
// Apply both transforms
|
||||||
|
this.canvas.style.transform = `translate(${currentX}px, ${currentY}px) scale(${this.scale})`;
|
||||||
|
this.canvas.style.transformOrigin = 'center center';
|
||||||
|
this.requestUpdate();
|
||||||
|
}
|
||||||
|
|
||||||
|
disconnectedCallback() {
|
||||||
|
super.disconnectedCallback();
|
||||||
|
this.removeEventListener('wheel', this._handleWheel);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user