257 lines
7.1 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';
2021-12-19 08:31:29 -08:00
import { Point, CurvedLine, Rect } from '@wisemapping/web2d';
2021-10-04 17:05:34 -07:00
import DragTopicConfig from './DragTopicConfig';
2022-02-23 16:12:43 -08:00
import SizeType from './SizeType';
import Topic from './Topic';
import Shape from './util/Shape';
2022-02-23 16:12:43 -08:00
import Workspace from './Workspace';
2021-10-04 17:05:34 -07:00
2021-12-04 15:39:20 -08:00
class DragPivot {
2022-02-23 16:12:43 -08:00
private _position: Point;
private _isVisible: boolean;
private _targetTopic: Topic;
private _connectRect: Rect;
private _dragPivot: Rect;
private _curvedLine: CurvedLine;
private _straightLine: CurvedLine;
private _size: SizeType;
2021-12-04 15:39:20 -08:00
constructor() {
2021-12-19 08:31:29 -08:00
this._position = new Point();
this._size = DragTopicConfig.PIVOT_SIZE;
2021-10-04 17:05:34 -07:00
this._straightLine = this._buildStraightLine();
this._curvedLine = this._buildCurvedLine();
this._dragPivot = this._buildRect();
this._connectRect = this._buildRect();
this._targetTopic = null;
this._isVisible = false;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
isVisible(): boolean {
2021-10-04 17:05:34 -07:00
return this._isVisible;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
getTargetTopic(): Topic {
2021-10-04 17:05:34 -07:00
return this._targetTopic;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
private _buildStraightLine(): CurvedLine {
2021-12-19 08:31:29 -08:00
const line = new CurvedLine();
line.setStyle(CurvedLine.SIMPLE_LINE);
2021-10-04 17:05:34 -07:00
line.setStroke(1, 'solid', '#CC0033');
line.setOpacity(0.4);
line.setVisibility(false);
return line;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
private _buildCurvedLine(): CurvedLine {
2021-12-19 08:31:29 -08:00
const line = new CurvedLine();
line.setStyle(CurvedLine.SIMPLE_LINE);
2021-10-04 17:05:34 -07:00
line.setStroke(1, 'solid', '#CC0033');
line.setOpacity(0.4);
line.setVisibility(false);
return line;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
private _redrawLine(): void {
2021-10-04 17:05:34 -07:00
// Update line position.
$assert(this.getTargetTopic(), 'Illegal invocation. Target node can not be null');
const pivotRect = this._getPivotRect();
// Pivot position has not changed. In this case, position change is not required.
const targetTopic = this.getTargetTopic();
const position = this._position;
// Calculate pivot connection point ...
const size = this._size;
const targetPosition = targetTopic.getPosition();
const line = this._getConnectionLine();
// Update Line position.
const isAtRight = Shape.isAtRight(targetPosition, position);
const pivotPoint = Shape.calculateRectConnectionPoint(position, size, isAtRight);
line.setFrom(pivotPoint.x, pivotPoint.y);
// Update rect position
2022-02-23 16:12:43 -08:00
const cx = position.x - size.width / 2;
const cy = position.y - size.height / 2;
2021-10-04 17:05:34 -07:00
pivotRect.setPosition(cx, cy);
// Make line visible only when the position has been already changed.
// This solve several strange effects ;)
const targetPoint = targetTopic.workoutIncomingConnectionPoint(pivotPoint);
line.setTo(targetPoint.x, targetPoint.y);
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
setPosition(point: Point): void {
2021-10-04 17:05:34 -07:00
this._position = point;
this._redrawLine();
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
getPosition(): Point {
2021-10-04 17:05:34 -07:00
return this._position;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
private _buildRect(): Rect {
2021-10-04 17:05:34 -07:00
const size = this._size;
const rectAttributes = {
fillColor: '#CC0033',
opacity: 0.4,
width: size.width,
height: size.height,
strokeColor: '#FF9933',
2021-10-04 17:05:34 -07:00
};
2021-12-19 08:31:29 -08:00
const rect = new Rect(0, rectAttributes);
2021-10-04 17:05:34 -07:00
rect.setVisibility(false);
return rect;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
private _getPivotRect(): Rect {
2021-10-04 17:05:34 -07:00
return this._dragPivot;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
getSize(): SizeType {
2021-10-04 17:05:34 -07:00
const elem2d = this._getPivotRect();
return elem2d.getSize();
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
setVisibility(value: boolean) {
if (this.isVisible() !== value) {
2021-10-04 17:05:34 -07:00
const pivotRect = this._getPivotRect();
pivotRect.setVisibility(value);
const connectRect = this._connectRect;
connectRect.setVisibility(value);
const line = this._getConnectionLine();
if (line) {
line.setVisibility(value);
}
this._isVisible = value;
}
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
// If the node is connected, validate that there is a line connecting both...
2022-02-23 16:12:43 -08:00
_getConnectionLine(): CurvedLine {
2021-10-04 17:05:34 -07:00
let result = null;
const parentTopic = this._targetTopic;
if (parentTopic) {
2022-01-02 14:16:18 -08:00
if (parentTopic.getType() === 'CentralTopic') {
2021-10-04 17:05:34 -07:00
result = this._straightLine;
} else {
result = this._curvedLine;
}
}
return result;
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
addToWorkspace(workspace: Workspace) {
2021-10-04 17:05:34 -07:00
const pivotRect = this._getPivotRect();
workspace.append(pivotRect);
const connectToRect = this._connectRect;
workspace.append(connectToRect);
// Add a hidden straight line ...
const straighLine = this._straightLine;
straighLine.setVisibility(false);
workspace.append(straighLine);
straighLine.moveToBack();
// Add a hidden curved line ...
const curvedLine = this._curvedLine;
curvedLine.setVisibility(false);
workspace.append(curvedLine);
curvedLine.moveToBack();
// Add a connect rect ...
const connectRect = this._connectRect;
connectRect.setVisibility(false);
workspace.append(connectRect);
connectRect.moveToBack();
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
removeFromWorkspace(workspace: Workspace) {
2021-10-04 17:05:34 -07:00
const shape = this._getPivotRect();
workspace.removeChild(shape);
const connectToRect = this._connectRect;
workspace.removeChild(connectToRect);
if ($defined(this._straightLine)) {
workspace.removeChild(this._straightLine);
}
if ($defined(this._curvedLine)) {
workspace.removeChild(this._curvedLine);
}
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
connectTo(targetTopic: Topic, position: Point) {
2021-10-04 17:05:34 -07:00
$assert(position, 'position can not be null');
$assert(targetTopic, 'parent can not be null');
this._position = position;
this._targetTopic = targetTopic;
// Connected to Rect ...
const connectRect = this._connectRect;
const targetSize = targetTopic.getSize();
// Add 4 pixel in order to keep create a rect bigger than the topic.
const width = targetSize.width + 4;
const height = targetSize.height + 4;
connectRect.setSize(width, height);
const targetPosition = targetTopic.getPosition();
const cx = Math.ceil(targetPosition.x - width / 2);
const cy = Math.ceil(targetPosition.y - height / 2);
2021-10-04 17:05:34 -07:00
connectRect.setPosition(cx, cy);
// Change elements position ...
const pivotRect = this._getPivotRect();
pivotRect.moveToFront();
pivotRect.setPosition(position.x, position.y);
this._redrawLine();
2021-12-04 15:39:20 -08:00
}
2021-10-04 17:05:34 -07:00
2022-02-23 16:12:43 -08:00
disconnect(workspace: Workspace): void {
2021-10-04 17:05:34 -07:00
$assert(workspace, 'workspace can not be null.');
$assert(this._targetTopic, 'There are not connected topic.');
this.setVisibility(false);
this._targetTopic = null;
2021-12-04 15:39:20 -08:00
}
}
2021-10-04 17:05:34 -07:00
export default DragPivot;