mirror of
https://bitbucket.org/wisemapping/wisemapping-frontend.git
synced 2025-04-03 19:53:20 +02:00
301 lines
8.9 KiB
TypeScript
301 lines
8.9 KiB
TypeScript
/*
|
|
* Copyright [2021] [wisemapping]
|
|
*
|
|
* 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.
|
|
*/
|
|
import $ from 'jquery';
|
|
import { $assert } from '@wisemapping/core-js';
|
|
import Keyboard from './Keyboard';
|
|
import { Designer } from '..';
|
|
import Topic from './Topic';
|
|
|
|
import initHotKeyPluggin from '../../../../libraries/jquery.hotkeys';
|
|
|
|
// Provides dispatcher of keyevents by key...
|
|
initHotKeyPluggin($);
|
|
|
|
export type EventCallback = (event?: Event) => void;
|
|
class DesignerKeyboard extends Keyboard {
|
|
// eslint-disable-next-line no-use-before-define
|
|
private static _instance: DesignerKeyboard;
|
|
|
|
private static _disabled: boolean;
|
|
|
|
private static excludeFromEditor = ['Enter', 'CapsLock', 'Escape', 'F1', 'F3', 'F4', 'F5', 'F6', 'F7', 'F8', 'F9', 'F10', 'F11', 'F12'];
|
|
|
|
constructor(designer: Designer) {
|
|
super();
|
|
$assert(designer, 'designer can not be null');
|
|
this._registerEvents(designer);
|
|
}
|
|
|
|
addShortcut(shortcuts: string[] | string, callback: EventCallback): void {
|
|
super.addShortcut(shortcuts, () => {
|
|
if (DesignerKeyboard.isDisabled()) {
|
|
return;
|
|
}
|
|
callback();
|
|
});
|
|
}
|
|
|
|
private _registerEvents(designer: Designer) {
|
|
// Try with the keyboard ..
|
|
const model = designer.getModel();
|
|
this.addShortcut(['backspace', 'del'], () => { designer.deleteSelectedEntities(); });
|
|
|
|
this.addShortcut('space', () => { designer.shrinkSelectedBranch(); });
|
|
|
|
this.addShortcut('f2', () => {
|
|
const node = model.selectedTopic();
|
|
if (node) {
|
|
node.showTextEditor(node.getText());
|
|
}
|
|
});
|
|
|
|
this.addShortcut(['insert', 'tab', 'meta+enter'], () => { designer.createChildForSelectedNode(); });
|
|
|
|
this.addShortcut('enter', () => { designer.createSiblingForSelectedNode(); });
|
|
|
|
this.addShortcut(['ctrl+z', 'meta+z'], () => { designer.undo(); });
|
|
|
|
this.addShortcut(['ctrl+shift+z', 'meta+shift+z'], () => { designer.redo(); });
|
|
|
|
this.addShortcut(['ctrl+c', 'meta+c'], () => { designer.copyToClipboard(); });
|
|
|
|
this.addShortcut(['ctrl+l', 'meta+l'], () => { designer.addLink(); });
|
|
|
|
this.addShortcut(['ctrl+k', 'meta+k'], () => { designer.addNote(); });
|
|
|
|
this.addShortcut(['ctrl+v', 'meta+v'], () => { designer.pasteClipboard(); });
|
|
|
|
this.addShortcut(['ctrl+a', 'meta+a'], () => { designer.selectAll(); });
|
|
|
|
this.addShortcut(['ctrl+b', 'meta+b'], () => { designer.changeFontWeight(); });
|
|
|
|
this.addShortcut(['ctrl+s', 'meta+s'], () => { $(document).find('#save').trigger('click'); });
|
|
|
|
this.addShortcut(['ctrl+i', 'meta+i'], () => { designer.changeFontStyle(); });
|
|
|
|
this.addShortcut(['ctrl+shift+a', 'meta+shift+a'], () => { designer.deselectAll(); });
|
|
|
|
this.addShortcut(['meta+=', 'ctrl+='], () => { designer.zoomIn(); });
|
|
|
|
this.addShortcut(['meta+-', 'ctrl+-'], () => { designer.zoomOut(); });
|
|
|
|
const me = this;
|
|
this.addShortcut(
|
|
'right', () => {
|
|
const node = model.selectedTopic();
|
|
if (node) {
|
|
if (node.isCentralTopic()) {
|
|
me._goToSideChild(designer, node, 'RIGHT');
|
|
} else if (node.getPosition().x < 0) {
|
|
me._goToParent(designer, node);
|
|
} else if (!node.areChildrenShrunken()) {
|
|
me._goToChild(designer, node);
|
|
}
|
|
} else {
|
|
const centralTopic = model.getCentralTopic();
|
|
me._goToNode(designer, centralTopic);
|
|
}
|
|
},
|
|
);
|
|
|
|
this.addShortcut(
|
|
'left', () => {
|
|
const node = model.selectedTopic();
|
|
if (node) {
|
|
if (node.isCentralTopic()) {
|
|
me._goToSideChild(designer, node, 'LEFT');
|
|
} else if (node.getPosition().x > 0) {
|
|
me._goToParent(designer, node);
|
|
} else if (!node.areChildrenShrunken()) {
|
|
me._goToChild(designer, node);
|
|
}
|
|
} else {
|
|
const centralTopic = model.getCentralTopic();
|
|
me._goToNode(designer, centralTopic);
|
|
}
|
|
},
|
|
);
|
|
|
|
this.addShortcut(
|
|
'up', () => {
|
|
const node = model.selectedTopic();
|
|
if (node) {
|
|
if (!node.isCentralTopic()) {
|
|
me._goToBrother(designer, node, 'UP');
|
|
}
|
|
} else {
|
|
const centralTopic = model.getCentralTopic();
|
|
me._goToNode(designer, centralTopic);
|
|
}
|
|
},
|
|
);
|
|
this.addShortcut(
|
|
'down', () => {
|
|
const node = model.selectedTopic();
|
|
if (node) {
|
|
if (!node.isCentralTopic()) {
|
|
me._goToBrother(designer, node, 'DOWN');
|
|
}
|
|
} else {
|
|
const centralTopic = model.getCentralTopic();
|
|
me._goToNode(designer, centralTopic);
|
|
}
|
|
},
|
|
);
|
|
|
|
$(document).on('keypress', (event) => {
|
|
// Needs to be ignored ?
|
|
if (DesignerKeyboard.isDisabled() || DesignerKeyboard.excludeFromEditor.includes(event.code)) {
|
|
return;
|
|
}
|
|
|
|
// Is a modifier ?
|
|
if (event.ctrlKey || event.altKey || event.metaKey) {
|
|
return;
|
|
}
|
|
|
|
// If a node is selected, open the editor ...
|
|
const topic = designer.getModel().selectedTopic();
|
|
if (topic) {
|
|
event.stopPropagation();
|
|
event.preventDefault();
|
|
topic.showTextEditor(event.key);
|
|
}
|
|
});
|
|
}
|
|
|
|
private _goToBrother(designer: Designer, node: Topic, direction) {
|
|
const parent = node.getParent();
|
|
if (parent) {
|
|
const brothers = parent.getChildren();
|
|
|
|
let target = node;
|
|
const { y } = node.getPosition();
|
|
const { x } = node.getPosition();
|
|
let dist = null;
|
|
for (let i = 0; i < brothers.length; i++) {
|
|
const sameSide = (x * brothers[i].getPosition().x) >= 0;
|
|
if (brothers[i] !== node && sameSide) {
|
|
const brother = brothers[i];
|
|
const brotherY = brother.getPosition().y;
|
|
if (direction === 'DOWN' && brotherY > y) {
|
|
let distancia = y - brotherY;
|
|
if (distancia < 0) {
|
|
distancia *= (-1);
|
|
}
|
|
if (dist == null || dist > distancia) {
|
|
dist = distancia;
|
|
target = brothers[i];
|
|
}
|
|
} else if (direction === 'UP' && brotherY < y) {
|
|
let distance = y - brotherY;
|
|
if (distance < 0) {
|
|
distance *= (-1);
|
|
}
|
|
if (dist == null || dist > distance) {
|
|
dist = distance;
|
|
target = brothers[i];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
this._goToNode(designer, target);
|
|
}
|
|
}
|
|
|
|
private _goToSideChild(designer: Designer, node: Topic, side: 'LEFT' | 'RIGHT') {
|
|
const children = node.getChildren();
|
|
if (children.length > 0) {
|
|
let target = children[0];
|
|
let top = null;
|
|
for (let i = 0; i < children.length; i++) {
|
|
const child = children[i];
|
|
const childY = child.getPosition().y;
|
|
if (side === 'LEFT' && child.getPosition().x < 0) {
|
|
if (top == null || childY < top) {
|
|
target = child;
|
|
top = childY;
|
|
}
|
|
}
|
|
if (side === 'RIGHT' && child.getPosition().x > 0) {
|
|
if (top == null || childY < top) {
|
|
target = child;
|
|
top = childY;
|
|
}
|
|
}
|
|
}
|
|
|
|
this._goToNode(designer, target);
|
|
}
|
|
}
|
|
|
|
private _goToParent(designer: Designer, node: Topic) {
|
|
const parent = node.getParent();
|
|
if (parent) {
|
|
this._goToNode(designer, parent);
|
|
}
|
|
}
|
|
|
|
private _goToChild(designer: Designer, node: Topic) {
|
|
const children = node.getChildren();
|
|
if (children.length > 0) {
|
|
let target = children[0];
|
|
let top = target.getPosition().y;
|
|
for (let i = 0; i < children.length; i++) {
|
|
const child = children[i];
|
|
if (child.getPosition().y < top) {
|
|
top = child.getPosition().y;
|
|
target = child;
|
|
}
|
|
}
|
|
this._goToNode(designer, target);
|
|
}
|
|
}
|
|
|
|
private _goToNode(designer: Designer, node: Topic) {
|
|
// First deselect all the nodes ...
|
|
designer.deselectAll();
|
|
|
|
// Give focus to the selected node....
|
|
node.setOnFocus(true);
|
|
}
|
|
|
|
static register = function register(designer: Designer) {
|
|
this._instance = new DesignerKeyboard(designer);
|
|
this._disabled = false;
|
|
};
|
|
|
|
static pause() {
|
|
this._disabled = true;
|
|
}
|
|
|
|
static resume() {
|
|
this._disabled = false;
|
|
}
|
|
|
|
static isDisabled() {
|
|
return this._disabled;
|
|
}
|
|
|
|
static getInstance() {
|
|
return this._instance;
|
|
}
|
|
}
|
|
|
|
export default DesignerKeyboard;
|