252 lines
7.7 KiB
TypeScript
Raw Normal View History

2021-10-04 17:05:34 -07:00
/*
2021-12-25 14:39:34 -08:00
* Copyright [2021] [wisemapping]
2021-10-04 17:05:34 -07:00
*
* Licensed under WiseMapping Public License, Version 1.0 (the "License").
* It is basically the Apache License, Version 2.0 (the "License") plus the
* "powered by wisemapping" text requirement on every single page;
* you may not use this file except in compliance with the License.
* You may obtain a copy of the license at
*
* http://www.wisemapping.org/license
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
2021-12-03 16:11:17 -08:00
import { $assert, $defined } from '@wisemapping/core-js';
import { Workspace as Workspace2D } from '@wisemapping/web2d';
2022-01-08 17:08:26 -08:00
import ScreenManager from './ScreenManager';
2021-10-04 17:05:34 -07:00
2021-12-03 10:58:25 -08:00
class Workspace {
2022-01-08 17:08:26 -08:00
_zoom: number;
_screenManager: ScreenManager;
_isReadOnly: boolean;
_screenWidth: number;
_screenHeight: number;
_workspace: Workspace2D;
_eventsEnabled: boolean;
_viewPort: { height: number, width: number };
constructor(screenManager: ScreenManager, zoom: number, isReadOnly: boolean) {
2021-10-04 17:05:34 -07:00
// Create a suitable container ...
$assert(screenManager, 'Div container can not be null');
$assert(zoom, 'zoom container can not be null');
this._zoom = zoom;
this._screenManager = screenManager;
this._isReadOnly = isReadOnly;
2021-10-04 17:05:34 -07:00
const divContainer = screenManager.getContainer();
this._screenWidth = Number.parseInt(divContainer.css('width'), 10);
this._screenHeight = Number.parseInt(divContainer.css('height'), 10);
2021-10-04 17:05:34 -07:00
// Initialize web2d workspace.
const workspace = this._createWorkspace();
this._workspace = workspace;
// Append to the workspace...
workspace.addItAsChildTo(divContainer);
this.setZoom(zoom, true);
// Register drag events ...
this._registerDragEvents();
this._eventsEnabled = true;
}
get isReadOnly() {
return this._isReadOnly;
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
private _createWorkspace() {
2021-10-04 17:05:34 -07:00
// Initialize workspace ...
const coordOriginX = -(this._screenWidth / 2);
const coordOriginY = -(this._screenHeight / 2);
const workspaceProfile = {
width: `${this._screenWidth}px`,
height: `${this._screenHeight}px`,
coordSizeWidth: this._screenWidth,
coordSizeHeight: this._screenHeight,
coordOriginX,
coordOriginY,
fillColor: 'transparent',
strokeWidth: 0,
};
2021-12-19 08:31:29 -08:00
return new Workspace2D(workspaceProfile);
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
append(shape) {
if ($defined(shape.addToWorkspace)) {
shape.addToWorkspace(this);
} else {
this._workspace.append(shape);
}
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
removeChild(shape) {
// Element is a node, not a web2d element?
if ($defined(shape.removeFromWorkspace)) {
shape.removeFromWorkspace(this);
} else {
this._workspace.removeChild(shape);
}
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
addEvent(type, listener) {
this._workspace.addEvent(type, listener);
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
removeEvent(type, listener) {
$assert(type, 'type can not be null');
$assert(listener, 'listener can not be null');
this._workspace.removeEvent(type, listener);
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
getSize() {
return this._workspace.getCoordSize();
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
setZoom(zoom: number, center: boolean = false): void {
2021-10-04 17:05:34 -07:00
this._zoom = zoom;
const workspace = this._workspace;
// Calculate the original boxview size ...
let origWidth = 0;
let origHeight = 0
if (this._viewPort) {
origWidth = this._viewPort.width;
origHeight = this._viewPort.height;
}
2021-10-04 17:05:34 -07:00
// Update coord scale...
const coordWidth = zoom * this._screenWidth;
const coordHeight = zoom * this._screenHeight;
workspace.setCoordSize(coordWidth, coordHeight);
// View port coords ...
if (this._viewPort) {
this._viewPort.width *= zoom;
this._viewPort.height *= zoom;
2021-10-04 17:05:34 -07:00
}
// Center topic....
let coordOriginX: number;
let coordOriginY: number;
2021-10-04 17:05:34 -07:00
if (center) {
if (this._viewPort) {
coordOriginX = -(this._viewPort.width / 2);
coordOriginY = -(this._viewPort.height / 2);
} else {
coordOriginX = -(coordWidth / 2);
coordOriginY = -(coordHeight / 2);
}
} else {
// Zoom keeping the center in the same place ...
// const xPaddinng = (this._viewPort.width - origWidth) / 2;
// const yPaddinng = (this._viewPort.height - origHeight) / 2;
// const coordOrigin = workspace.getCoordOrigin();
// coordOriginX = coordOrigin.x - xPaddinng;
// coordOriginY = coordOrigin.y - yPaddinng;
2021-10-04 17:05:34 -07:00
const coordOrigin = workspace.getCoordOrigin();
coordOriginX = coordOrigin.x;
coordOriginY = coordOrigin.y;
2021-10-04 17:05:34 -07:00
}
2021-10-04 17:05:34 -07:00
workspace.setCoordOrigin(coordOriginX, coordOriginY);
// Update screen.
this._screenManager.setOffset(coordOriginX, coordOriginY);
this._screenManager.setScale(zoom);
// Some changes in the screen. Let's fire an update event...
this._screenManager.fireEvent('update');
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
2022-01-08 17:08:26 -08:00
getScreenManager(): ScreenManager {
2021-10-04 17:05:34 -07:00
return this._screenManager;
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
2022-01-08 17:08:26 -08:00
enableWorkspaceEvents(value: boolean) {
2021-10-04 17:05:34 -07:00
this._eventsEnabled = value;
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
2022-01-08 17:08:26 -08:00
isWorkspaceEventsEnabled(): boolean {
2021-10-04 17:05:34 -07:00
return this._eventsEnabled;
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
getSVGElement() {
return this._workspace.getSVGElement();
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
2022-01-08 17:08:26 -08:00
private _registerDragEvents() {
2021-10-04 17:05:34 -07:00
const workspace = this._workspace;
const screenManager = this._screenManager;
const mWorkspace = this;
2021-12-19 08:31:29 -08:00
const mouseDownListener = function mouseDownListener(event) {
2021-10-04 17:05:34 -07:00
if (!$defined(workspace._mouseMoveListener)) {
if (mWorkspace.isWorkspaceEventsEnabled()) {
mWorkspace.enableWorkspaceEvents(false);
const mouseDownPosition = screenManager.getWorkspaceMousePosition(event);
const originalCoordOrigin = workspace.getCoordOrigin();
let wasDragged = false;
workspace._mouseMoveListener = (mouseMoveEvent) => {
const currentMousePosition = screenManager.getWorkspaceMousePosition(mouseMoveEvent);
2021-10-04 17:05:34 -07:00
const offsetX = currentMousePosition.x - mouseDownPosition.x;
const coordOriginX = -offsetX + originalCoordOrigin.x;
const offsetY = currentMousePosition.y - mouseDownPosition.y;
const coordOriginY = -offsetY + originalCoordOrigin.y;
workspace.setCoordOrigin(coordOriginX, coordOriginY);
// Change cursor.
2021-12-23 09:56:36 -08:00
window.document.body.style.cursor = 'move';
mouseMoveEvent.preventDefault();
2021-10-04 17:05:34 -07:00
// Fire drag event ...
screenManager.fireEvent('update');
wasDragged = true;
};
screenManager.addEvent('mousemove', workspace._mouseMoveListener);
// Register mouse up listeners ...
workspace._mouseUpListener = () => {
2021-10-04 17:05:34 -07:00
screenManager.removeEvent('mousemove', workspace._mouseMoveListener);
screenManager.removeEvent('mouseup', workspace._mouseUpListener);
workspace._mouseUpListener = null;
workspace._mouseMoveListener = null;
window.document.body.style.cursor = 'default';
// Update screen manager offset.
const coordOrigin = workspace.getCoordOrigin();
screenManager.setOffset(coordOrigin.x, coordOrigin.y);
mWorkspace.enableWorkspaceEvents(true);
if (!wasDragged) {
screenManager.fireEvent('click');
}
};
screenManager.addEvent('mouseup', workspace._mouseUpListener);
}
} else {
workspace._mouseUpListener();
}
};
screenManager.addEvent('mousedown', mouseDownListener);
2021-12-03 10:58:25 -08:00
}
2021-10-04 17:05:34 -07:00
setViewPort(size: { height: number, width: number }) {
2021-10-04 17:05:34 -07:00
this._viewPort = size;
2021-12-03 10:58:25 -08:00
}
}
2021-10-04 17:05:34 -07:00
export default Workspace;