blob: a205613b005488598d60ee18813ffdb5163e3de9 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// NOTE: This file depends on ui.js (or the autogenerated ui.m.js module
// version). These files and all files that depend on them are deprecated, and
// should only be used by legacy UIs that have not yet been updated to new
// patterns. Use Web Components in any new code.
/**
* @fileoverview A command is an abstraction of an action a user can do in the
* UI.
*
* When the focus changes in the document for each command a canExecute event
* is dispatched on the active element. By listening to this event you can
* enable and disable the command by setting the event.canExecute property.
*
* When a command is executed a command event is dispatched on the active
* element. Note that you should stop the propagation after you have handled the
* command if there might be other command listeners higher up in the DOM tree.
*/
// clang-format off
import {assert} from 'chrome://resources/js/assert_ts.js';
import {define as crUiDefine} from './ui.js';
import {KeyboardShortcutList} from 'chrome://resources/js/keyboard_shortcut_list.js';
import {dispatchPropertyChange, getPropertyDescriptor, PropertyKind} from './cr_deprecated.js';
import {MenuItem} from './menu_item.js';
// clang-format on
/**
* Creates a new command element.
* @constructor
* @extends {HTMLElement}
*/
export const Command = crUiDefine('command');
Command.prototype = {
__proto__: HTMLElement.prototype,
/**
* Initializes the command.
*/
decorate() {
assert(this.ownerDocument);
CommandManager.init(this.ownerDocument);
if (this.hasAttribute('shortcut')) {
this.shortcut = this.getAttribute('shortcut');
}
},
/**
* Executes the command by dispatching a command event on the given element.
* If |element| isn't given, the active element is used instead.
* If the command is {@code disabled} this does nothing.
* @param {HTMLElement=} opt_element Optional element to dispatch event on.
*/
execute(opt_element) {
if (this.disabled) {
return;
}
const doc = this.ownerDocument;
if (doc.activeElement) {
const e = new Event('command', {bubbles: true});
e.command = this;
(opt_element || doc.activeElement).dispatchEvent(e);
}
},
/**
* Call this when there have been changes that might change whether the
* command can be executed or not.
* @param {Node=} opt_node Node for which to actuate command state.
*/
canExecuteChange(opt_node) {
dispatchCanExecuteEvent(this, opt_node || this.ownerDocument.activeElement);
},
/**
* The keyboard shortcut that triggers the command. This is a string
* consisting of a key (as reported by WebKit in keydown) as
* well as optional key modifiers joinded with a '|'.
*
* Multiple keyboard shortcuts can be provided by separating them by
* whitespace.
*
* For example:
* "F1"
* "Backspace|Meta" for Apple command backspace.
* "a|Ctrl" for Control A
* "Delete Backspace|Meta" for Delete and Command Backspace
*
* @type {string}
*/
shortcut_: '',
get shortcut() {
return this.shortcut_;
},
set shortcut(shortcut) {
const oldShortcut = this.shortcut_;
if (shortcut !== oldShortcut) {
this.keyboardShortcuts_ = new KeyboardShortcutList(shortcut);
// Set this after the keyboardShortcuts_ since that might throw.
this.shortcut_ = shortcut;
dispatchPropertyChange(this, 'shortcut', this.shortcut_, oldShortcut);
}
},
/**
* Whether the event object matches the shortcut for this command.
* @param {!Event} e The key event object.
* @return {boolean} Whether it matched or not.
*/
matchesEvent(e) {
if (!this.keyboardShortcuts_) {
return false;
}
return this.keyboardShortcuts_.matchesEvent(e);
},
};
/**
* The label of the command.
* @type {string}
*/
Command.prototype.label;
Object.defineProperty(
Command.prototype, 'label',
getPropertyDescriptor('label', PropertyKind.ATTR));
/**
* Whether the command is disabled or not.
* @type {boolean}
*/
Command.prototype.disabled;
Object.defineProperty(
Command.prototype, 'disabled',
getPropertyDescriptor('disabled', PropertyKind.BOOL_ATTR));
/**
* Whether the command is hidden or not.
*/
Object.defineProperty(
Command.prototype, 'hidden',
getPropertyDescriptor('hidden', PropertyKind.BOOL_ATTR));
/**
* Whether the command is checked or not.
* @type {boolean}
*/
Command.prototype.checked;
Object.defineProperty(
Command.prototype, 'checked',
getPropertyDescriptor('checked', PropertyKind.BOOL_ATTR));
/**
* The flag that prevents the shortcut text from being displayed on menu.
*
* If false, the keyboard shortcut text (eg. "Ctrl+X" for the cut command)
* is displayed in menu when the command is associated with a menu item.
* Otherwise, no text is displayed.
* @type {boolean}
*/
Command.prototype.hideShortcutText;
Object.defineProperty(
Command.prototype, 'hideShortcutText',
getPropertyDescriptor('hideShortcutText', PropertyKind.BOOL_ATTR));
/**
* Dispatches a canExecute event on the target.
* @param {!Command} command The command that we are testing for.
* @param {EventTarget} target The target element to dispatch the event on.
*/
function dispatchCanExecuteEvent(command, target) {
const e = new CanExecuteEvent(command);
target.dispatchEvent(e);
command.disabled = !e.canExecute;
}
/**
* The command managers for different documents.
* @type {!Map<!Document, !CommandManager>}
*/
const commandManagers = new Map();
/**
* Keeps track of the focused element and updates the commands when the focus
* changes.
* @param {!Document} doc The document that we are managing the commands for.
* @constructor
*/
function CommandManager(doc) {
doc.addEventListener('focus', this.handleFocus_.bind(this), true);
// Make sure we add the listener to the bubbling phase so that elements can
// prevent the command.
doc.addEventListener('keydown', this.handleKeyDown_.bind(this), false);
}
/**
* Initializes a command manager for the document as needed.
* @param {!Document} doc The document to manage the commands for.
*/
CommandManager.init = function(doc) {
if (!commandManagers.has(doc)) {
commandManagers.set(doc, new CommandManager(doc));
}
};
CommandManager.prototype = {
/**
* Handles focus changes on the document.
* @param {Event} e The focus event object.
* @private
* @suppress {checkTypes}
* TODO(vitalyp): remove the suppression.
*/
handleFocus_(e) {
const target = e.target;
// Ignore focus on a menu button or command item.
if (target.menu || target.command || (target instanceof MenuItem)) {
return;
}
const commands = Array.prototype.slice.call(
target.ownerDocument.querySelectorAll('command'));
commands.forEach(function(command) {
dispatchCanExecuteEvent(command, target);
});
},
/**
* Handles the keydown event and routes it to the right command.
* @param {!Event} e The keydown event.
*/
handleKeyDown_(e) {
const target = e.target;
const commands = Array.prototype.slice.call(
target.ownerDocument.querySelectorAll('command'));
for (let i = 0, command; command = commands[i]; i++) {
if (command.matchesEvent(e)) {
// When invoking a command via a shortcut, we have to manually check
// if it can be executed, since focus might not have been changed
// what would have updated the command's state.
command.canExecuteChange();
if (!command.disabled) {
e.preventDefault();
// We do not want any other element to handle this.
e.stopPropagation();
command.execute();
return;
}
}
}
},
};
/**
* The event type used for canExecute events.
* @param {!Command} command The command that we are evaluating.
* @extends {Event}
* @constructor
* @class
*/
export function CanExecuteEvent(command) {
const e = new Event('canExecute', {bubbles: true, cancelable: true});
e.__proto__ = CanExecuteEvent.prototype;
e.command = command;
return e;
}
CanExecuteEvent.prototype = {
__proto__: Event.prototype,
/**
* The current command
* @type {Command}
*/
command: null,
/**
* Whether the target can execute the command. Setting this also stops the
* propagation and prevents the default. Callers can tell if an event has
* been handled via |this.defaultPrevented|.
* @type {boolean}
*/
canExecute_: false,
get canExecute() {
return this.canExecute_;
},
set canExecute(canExecute) {
this.canExecute_ = !!canExecute;
this.stopPropagation();
this.preventDefault();
},
};