UNPKG

@jupyterlab/apputils

Version:
292 lines 10.2 kB
/* ----------------------------------------------------------------------------- | Copyright (c) Jupyter Development Team. | Distributed under the terms of the Modified BSD License. |----------------------------------------------------------------------------*/ import { JSONExt } from '@lumino/coreutils'; import { Dialog } from './dialog'; import { showCommandLinkerTrustDialog } from './commandlinkertrustdialog'; /** * The command data attribute added to nodes that are connected. */ const COMMAND_ATTR = 'commandlinker-command'; /** * The args data attribute added to nodes that are connected. */ const ARGS_ATTR = 'commandlinker-args'; /** * The trust command data attribute added to nodes that are connected. */ const TRUST_COMMAND_ATTR = 'trust-command'; /** * A static class that provides helper methods to generate clickable nodes that * execute registered commands with pre-populated arguments. */ export class CommandLinker { /** * Instantiate a new command linker. */ constructor(options) { this._isDisposed = false; this._trustedBoundaries = new WeakMap(); this._commands = options.commands; document.body.addEventListener('click', this); } /** * Test whether the linker is disposed. */ get isDisposed() { return this._isDisposed; } /** * Dispose of the resources held by the linker. */ dispose() { if (this.isDisposed) { return; } this._isDisposed = true; document.body.removeEventListener('click', this); } /** * Connect a command/argument pair to a given node so that when it is clicked, * the command will execute. * * @param node - The node being connected. * * @param command - The command ID to execute upon click. * * @param args - The arguments with which to invoke the command. * * @returns The same node that was passed in, after it has been connected. * * #### Notes * Only `click` events will execute the command on a connected node. So, there * are two considerations that are relevant: * 1. If a node is connected, the default click action will be prevented. * 2. The `HTMLElement` passed in should be clickable. */ connectNode(node, command, args) { node.setAttribute(`data-${COMMAND_ATTR}`, command); if (args !== void 0) { node.setAttribute(`data-${ARGS_ATTR}`, JSON.stringify(args)); } return node; } /** * Mark a node as trusted for command execution. * * #### Notes * This trust marker is kept in-memory and cannot be forged from untrusted * DOM content. The mark applies to the node and all of its descendants. */ markTrusted(node) { this._trustedBoundaries.set(node, true); return node; } /** * Remove the trusted boundary marker from a node. * * #### Notes * This method is safe to call on nodes that were never marked as trusted. */ unmarkTrusted(node) { this._trustedBoundaries.delete(node); return node; } /** * Disconnect a node that has been connected to execute a command on click. * * @param node - The node being disconnected. * * @returns The same node that was passed in, after it has been disconnected. * * #### Notes * This method is safe to call multiple times and is safe to call on nodes * that were never connected. * * This method can be called on rendered virtual DOM nodes that were populated * using the `populateVNodeDataset` method in order to disconnect them from * executing their command/argument pair. */ disconnectNode(node) { node.removeAttribute(`data-${COMMAND_ATTR}`); node.removeAttribute(`data-${ARGS_ATTR}`); return node; } /** * Handle the DOM events for the command linker helper class. * * @param event - The DOM event sent to the class. * * #### Notes * This method implements the DOM `EventListener` interface and is * called in response to events on the panel's DOM node. It should * not be called directly by user code. */ handleEvent(event) { switch (event.type) { case 'click': this._evtClick(event).catch(console.warn); break; default: return; } } /** * Populate the `dataset` attribute within the collection of attributes used * to instantiate a virtual DOM node with the values necessary for its * rendered DOM node to respond to clicks by executing a command/argument * pair. * * @param command - The command ID to execute upon click. * * @param args - The arguments with which to invoke the command. * * @returns A `dataset` collection for use within virtual node attributes. * * #### Notes * The return value can be used on its own as the value for the `dataset` * attribute of a virtual element, or it can be added to an existing `dataset` * as in the example below. * * #### Example * ```typescript * let command = 'some:command-id'; * let args = { alpha: 'beta' }; * let anchor = h.a({ * className: 'some-class', * dataset: { * foo: '1', * bar: '2', * ../...linker.populateVNodeDataset(command, args) * } * }, 'some text'); * ``` */ populateVNodeDataset(command, args) { let dataset; if (args !== void 0) { dataset = { [ARGS_ATTR]: JSON.stringify(args), [COMMAND_ATTR]: command }; } else { dataset = { [COMMAND_ATTR]: command }; } return dataset; } /** * The global click handler that deploys commands/argument pairs that are * attached to the node being clicked. */ async _evtClick(event) { let target = event.target; while (target && target.parentElement) { if (target.hasAttribute(`data-${COMMAND_ATTR}`)) { event.preventDefault(); const command = target.getAttribute(`data-${COMMAND_ATTR}`); if (!command) { return; } const argsValue = target.getAttribute(`data-${ARGS_ATTR}`); let args = JSONExt.emptyObject; if (argsValue) { args = JSON.parse(argsValue); } if (!this._isTrusted(target)) { const label = this._commands.label(command, args) || command; const trustCommand = this._findTrustCommand(target); const trustAction = await this._requestTrust({ args, command, label, trustCommand }); if (trustAction === 'cancel') { return; } if (trustAction === 'trust' && trustCommand) { try { const trustResult = await this._commands.execute(trustCommand); if (!Private.isTrustCommandResult(trustResult) || !trustResult.trusted) { return; } } catch (error) { console.error('Failed to execute trust command', error); return; } } } void this._commands.execute(command, args); return; } target = target.parentElement; } } _isTrusted(target) { let node = target; while (node) { if (this._trustedBoundaries.has(node)) { return true; } node = node.parentElement; } return false; } _findTrustCommand(target) { var _a, _b; const trustNode = (_a = target.parentElement) === null || _a === void 0 ? void 0 : _a.closest(`[data-${TRUST_COMMAND_ATTR}]`); return (_b = trustNode === null || trustNode === void 0 ? void 0 : trustNode.getAttribute(`data-${TRUST_COMMAND_ATTR}`)) !== null && _b !== void 0 ? _b : null; } async _requestTrust(request) { const trans = Dialog.translator.load('jupyterlab'); const args = JSON.stringify(request.args, null, 2); const buttons = [ Dialog.cancelButton({ label: trans.__('Cancel') }), Dialog.okButton({ label: request.trustCommand ? trans.__('Run Once') : trans.__('Run'), actions: ['run'] }) ]; if (request.trustCommand) { buttons.push(Dialog.warnButton({ label: trans.__('Trust'), accept: false, actions: ['trust'] })); } const result = await showCommandLinkerTrustDialog({ args, buttons, command: request.command, commandLabel: request.label, hasTrustCommand: !!request.trustCommand, defaultButton: 0, title: trans.__('Run Command?') }); if (result.button.actions.includes('trust')) { return 'trust'; } if (result.button.actions.includes('run') || result.button.accept) { return 'run'; } return 'cancel'; } } /** * A namespace for Private command linker statics. */ var Private; (function (Private) { /** * Type guard for trust command results. */ function isTrustCommandResult(result) { return (typeof result === 'object' && result !== null && 'trusted' in result && typeof result.trusted === 'boolean'); } Private.isTrustCommandResult = isTrustCommandResult; })(Private || (Private = {})); //# sourceMappingURL=commandlinker.js.map