Files
blockly/core/shortcut_items.js
2021-10-21 09:01:51 -07:00

303 lines
9.6 KiB
JavaScript

/**
* @license
* Copyright 2020 Google LLC
* SPDX-License-Identifier: Apache-2.0
*/
/**
* @fileoverview Registers default keyboard shortcuts.
*/
'use strict';
/**
* Registers default keyboard shortcuts.
* @namespace Blockly.ShortcutItems
*/
goog.module('Blockly.ShortcutItems');
const clipboard = goog.require('Blockly.clipboard');
const common = goog.require('Blockly.common');
/* eslint-disable-next-line no-unused-vars */
const {BlockSvg} = goog.requireType('Blockly.BlockSvg');
const {Gesture} = goog.require('Blockly.Gesture');
/* eslint-disable-next-line no-unused-vars */
const {ICopyable} = goog.requireType('Blockly.ICopyable');
const {KeyCodes} = goog.require('Blockly.utils.KeyCodes');
const {ShortcutRegistry} = goog.require('Blockly.ShortcutRegistry');
/**
* Object holding the names of the default shortcut items.
* @enum {string}
* @alias Blockly.ShortcutItems.names
*/
const names = {
ESCAPE: 'escape',
DELETE: 'delete',
COPY: 'copy',
CUT: 'cut',
PASTE: 'paste',
UNDO: 'undo',
REDO: 'redo',
};
exports.names = names;
/**
* Keyboard shortcut to hide chaff on escape.
* @alias Blockly.ShortcutItems.registerEscape
*/
const registerEscape = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const escapeAction = {
name: names.ESCAPE,
preconditionFn: function(workspace) {
return !workspace.options.readOnly;
},
callback: function(workspace) {
workspace.hideChaff();
return true;
},
};
ShortcutRegistry.registry.register(escapeAction);
ShortcutRegistry.registry.addKeyMapping(KeyCodes.ESC, escapeAction.name);
};
exports.registerEscape = registerEscape;
/**
* Keyboard shortcut to delete a block on delete or backspace
* @alias Blockly.ShortcutItems.registerDelete
*/
const registerDelete = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const deleteShortcut = {
name: names.DELETE,
preconditionFn: function(workspace) {
const selected = common.getSelected();
return !workspace.options.readOnly && selected && selected.isDeletable();
},
callback: function(workspace, e) {
// Delete or backspace.
// Stop the browser from going back to the previous page.
// Do this first to prevent an error in the delete code from resulting in
// data loss.
e.preventDefault();
// Don't delete while dragging. Jeez.
if (Gesture.inProgress()) {
return false;
}
(/** @type {!BlockSvg} */ (common.getSelected())).checkAndDelete();
return true;
},
};
ShortcutRegistry.registry.register(deleteShortcut);
ShortcutRegistry.registry.addKeyMapping(KeyCodes.DELETE, deleteShortcut.name);
ShortcutRegistry.registry.addKeyMapping(
KeyCodes.BACKSPACE, deleteShortcut.name);
};
exports.registerDelete = registerDelete;
/**
* Keyboard shortcut to copy a block on ctrl+c, cmd+c, or alt+c.
* @alias Blockly.ShortcutItems.registerCopy
*/
const registerCopy = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const copyShortcut = {
name: names.COPY,
preconditionFn: function(workspace) {
const selected = common.getSelected();
return !workspace.options.readOnly && !Gesture.inProgress() && selected &&
selected.isDeletable() && selected.isMovable();
},
callback: function(workspace, e) {
// Prevent the default copy behavior, which may beep or otherwise indicate
// an error due to the lack of a selection.
e.preventDefault();
workspace.hideChaff();
clipboard.copy(/** @type {!ICopyable} */ (common.getSelected()));
return true;
},
};
ShortcutRegistry.registry.register(copyShortcut);
const ctrlC = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.C, [KeyCodes.CTRL]);
ShortcutRegistry.registry.addKeyMapping(ctrlC, copyShortcut.name);
const altC =
ShortcutRegistry.registry.createSerializedKey(KeyCodes.C, [KeyCodes.ALT]);
ShortcutRegistry.registry.addKeyMapping(altC, copyShortcut.name);
const metaC = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.C, [KeyCodes.META]);
ShortcutRegistry.registry.addKeyMapping(metaC, copyShortcut.name);
};
exports.registerCopy = registerCopy;
/**
* Keyboard shortcut to copy and delete a block on ctrl+x, cmd+x, or alt+x.
* @alias Blockly.ShortcutItems.registerCut
*/
const registerCut = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const cutShortcut = {
name: names.CUT,
preconditionFn: function(workspace) {
const selected = common.getSelected();
return !workspace.options.readOnly && !Gesture.inProgress() && selected &&
selected.isDeletable() && selected.isMovable() &&
!selected.workspace.isFlyout;
},
callback: function() {
const selected = common.getSelected();
if (!selected) {
// Shouldn't happen but appeases the type system
return false;
}
clipboard.copy(selected);
(/** @type {!BlockSvg} */ (selected)).checkAndDelete();
return true;
},
};
ShortcutRegistry.registry.register(cutShortcut);
const ctrlX = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.X, [KeyCodes.CTRL]);
ShortcutRegistry.registry.addKeyMapping(ctrlX, cutShortcut.name);
const altX =
ShortcutRegistry.registry.createSerializedKey(KeyCodes.X, [KeyCodes.ALT]);
ShortcutRegistry.registry.addKeyMapping(altX, cutShortcut.name);
const metaX = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.X, [KeyCodes.META]);
ShortcutRegistry.registry.addKeyMapping(metaX, cutShortcut.name);
};
exports.registerCut = registerCut;
/**
* Keyboard shortcut to paste a block on ctrl+v, cmd+v, or alt+v.
* @alias Blockly.ShortcutItems.registerPaste
*/
const registerPaste = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const pasteShortcut = {
name: names.PASTE,
preconditionFn: function(workspace) {
return !workspace.options.readOnly && !Gesture.inProgress();
},
callback: function() {
return clipboard.paste();
},
};
ShortcutRegistry.registry.register(pasteShortcut);
const ctrlV = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.V, [KeyCodes.CTRL]);
ShortcutRegistry.registry.addKeyMapping(ctrlV, pasteShortcut.name);
const altV =
ShortcutRegistry.registry.createSerializedKey(KeyCodes.V, [KeyCodes.ALT]);
ShortcutRegistry.registry.addKeyMapping(altV, pasteShortcut.name);
const metaV = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.V, [KeyCodes.META]);
ShortcutRegistry.registry.addKeyMapping(metaV, pasteShortcut.name);
};
exports.registerPaste = registerPaste;
/**
* Keyboard shortcut to undo the previous action on ctrl+z, cmd+z, or alt+z.
* @alias Blockly.ShortcutItems.registerUndo
*/
const registerUndo = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const undoShortcut = {
name: names.UNDO,
preconditionFn: function(workspace) {
return !workspace.options.readOnly && !Gesture.inProgress();
},
callback: function(workspace) {
// 'z' for undo 'Z' is for redo.
workspace.hideChaff();
workspace.undo(false);
return true;
},
};
ShortcutRegistry.registry.register(undoShortcut);
const ctrlZ = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.Z, [KeyCodes.CTRL]);
ShortcutRegistry.registry.addKeyMapping(ctrlZ, undoShortcut.name);
const altZ =
ShortcutRegistry.registry.createSerializedKey(KeyCodes.Z, [KeyCodes.ALT]);
ShortcutRegistry.registry.addKeyMapping(altZ, undoShortcut.name);
const metaZ = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.Z, [KeyCodes.META]);
ShortcutRegistry.registry.addKeyMapping(metaZ, undoShortcut.name);
};
exports.registerUndo = registerUndo;
/**
* Keyboard shortcut to redo the previous action on ctrl+shift+z, cmd+shift+z,
* or alt+shift+z.
* @alias Blockly.ShortcutItems.registerRedo
*/
const registerRedo = function() {
/** @type {!ShortcutRegistry.KeyboardShortcut} */
const redoShortcut = {
name: names.REDO,
preconditionFn: function(workspace) {
return !Gesture.inProgress() && !workspace.options.readOnly;
},
callback: function(workspace) {
// 'z' for undo 'Z' is for redo.
workspace.hideChaff();
workspace.undo(true);
return true;
},
};
ShortcutRegistry.registry.register(redoShortcut);
const ctrlShiftZ = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.Z, [KeyCodes.SHIFT, KeyCodes.CTRL]);
ShortcutRegistry.registry.addKeyMapping(ctrlShiftZ, redoShortcut.name);
const altShiftZ = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.Z, [KeyCodes.SHIFT, KeyCodes.ALT]);
ShortcutRegistry.registry.addKeyMapping(altShiftZ, redoShortcut.name);
const metaShiftZ = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.Z, [KeyCodes.SHIFT, KeyCodes.META]);
ShortcutRegistry.registry.addKeyMapping(metaShiftZ, redoShortcut.name);
// Ctrl-y is redo in Windows. Command-y is never valid on Macs.
const ctrlY = ShortcutRegistry.registry.createSerializedKey(
KeyCodes.Y, [KeyCodes.CTRL]);
ShortcutRegistry.registry.addKeyMapping(ctrlY, redoShortcut.name);
};
exports.registerRedo = registerRedo;
/**
* Registers all default keyboard shortcut item. This should be called once per
* instance of KeyboardShortcutRegistry.
* @alias Blockly.ShortcutItems.registerDefaultShortcuts
* @package
*/
const registerDefaultShortcuts = function() {
registerEscape();
registerDelete();
registerCopy();
registerCut();
registerPaste();
registerUndo();
registerRedo();
};
exports.registerDefaultShortcuts = registerDefaultShortcuts;
registerDefaultShortcuts();