1 | /**
|
2 | * @license
|
3 | * Copyright Google LLC All Rights Reserved.
|
4 | *
|
5 | * Use of this source code is governed by an MIT-style license that can be
|
6 | * found in the LICENSE file at https://angular.io/license
|
7 | */
|
8 | import { DOCUMENT, ɵgetDOM as getDOM } from '@angular/common';
|
9 | import { Inject, Injectable } from '@angular/core';
|
10 | import { EventManagerPlugin } from './event_manager';
|
11 | import * as i0 from "@angular/core";
|
12 | /**
|
13 | * Defines supported modifiers for key events.
|
14 | */
|
15 | const MODIFIER_KEYS = ['alt', 'control', 'meta', 'shift'];
|
16 | const DOM_KEY_LOCATION_NUMPAD = 3;
|
17 | // Map to convert some key or keyIdentifier values to what will be returned by getEventKey
|
18 | const _keyMap = {
|
19 | // The following values are here for cross-browser compatibility and to match the W3C standard
|
20 | // cf https://www.w3.org/TR/DOM-Level-3-Events-key/
|
21 | '\b': 'Backspace',
|
22 | '\t': 'Tab',
|
23 | '\x7F': 'Delete',
|
24 | '\x1B': 'Escape',
|
25 | 'Del': 'Delete',
|
26 | 'Esc': 'Escape',
|
27 | 'Left': 'ArrowLeft',
|
28 | 'Right': 'ArrowRight',
|
29 | 'Up': 'ArrowUp',
|
30 | 'Down': 'ArrowDown',
|
31 | 'Menu': 'ContextMenu',
|
32 | 'Scroll': 'ScrollLock',
|
33 | 'Win': 'OS'
|
34 | };
|
35 | // There is a bug in Chrome for numeric keypad keys:
|
36 | // https://code.google.com/p/chromium/issues/detail?id=155654
|
37 | // 1, 2, 3 ... are reported as A, B, C ...
|
38 | const _chromeNumKeyPadMap = {
|
39 | 'A': '1',
|
40 | 'B': '2',
|
41 | 'C': '3',
|
42 | 'D': '4',
|
43 | 'E': '5',
|
44 | 'F': '6',
|
45 | 'G': '7',
|
46 | 'H': '8',
|
47 | 'I': '9',
|
48 | 'J': '*',
|
49 | 'K': '+',
|
50 | 'M': '-',
|
51 | 'N': '.',
|
52 | 'O': '/',
|
53 | '\x60': '0',
|
54 | '\x90': 'NumLock'
|
55 | };
|
56 | /**
|
57 | * Retrieves modifiers from key-event objects.
|
58 | */
|
59 | const MODIFIER_KEY_GETTERS = {
|
60 | 'alt': (event) => event.altKey,
|
61 | 'control': (event) => event.ctrlKey,
|
62 | 'meta': (event) => event.metaKey,
|
63 | 'shift': (event) => event.shiftKey
|
64 | };
|
65 | /**
|
66 | * @publicApi
|
67 | * A browser plug-in that provides support for handling of key events in Angular.
|
68 | */
|
69 | export class KeyEventsPlugin extends EventManagerPlugin {
|
70 | /**
|
71 | * Initializes an instance of the browser plug-in.
|
72 | * @param doc The document in which key events will be detected.
|
73 | */
|
74 | constructor(doc) {
|
75 | super(doc);
|
76 | }
|
77 | /**
|
78 | * Reports whether a named key event is supported.
|
79 | * @param eventName The event name to query.
|
80 | * @return True if the named key event is supported.
|
81 | */
|
82 | supports(eventName) {
|
83 | return KeyEventsPlugin.parseEventName(eventName) != null;
|
84 | }
|
85 | /**
|
86 | * Registers a handler for a specific element and key event.
|
87 | * @param element The HTML element to receive event notifications.
|
88 | * @param eventName The name of the key event to listen for.
|
89 | * @param handler A function to call when the notification occurs. Receives the
|
90 | * event object as an argument.
|
91 | * @returns The key event that was registered.
|
92 | */
|
93 | addEventListener(element, eventName, handler) {
|
94 | const parsedEvent = KeyEventsPlugin.parseEventName(eventName);
|
95 | const outsideHandler = KeyEventsPlugin.eventCallback(parsedEvent['fullKey'], handler, this.manager.getZone());
|
96 | return this.manager.getZone().runOutsideAngular(() => {
|
97 | return getDOM().onAndCancel(element, parsedEvent['domEventName'], outsideHandler);
|
98 | });
|
99 | }
|
100 | static parseEventName(eventName) {
|
101 | const parts = eventName.toLowerCase().split('.');
|
102 | const domEventName = parts.shift();
|
103 | if ((parts.length === 0) || !(domEventName === 'keydown' || domEventName === 'keyup')) {
|
104 | return null;
|
105 | }
|
106 | const key = KeyEventsPlugin._normalizeKey(parts.pop());
|
107 | let fullKey = '';
|
108 | MODIFIER_KEYS.forEach(modifierName => {
|
109 | const index = parts.indexOf(modifierName);
|
110 | if (index > -1) {
|
111 | parts.splice(index, 1);
|
112 | fullKey += modifierName + '.';
|
113 | }
|
114 | });
|
115 | fullKey += key;
|
116 | if (parts.length != 0 || key.length === 0) {
|
117 | // returning null instead of throwing to let another plugin process the event
|
118 | return null;
|
119 | }
|
120 | // NOTE: Please don't rewrite this as so, as it will break JSCompiler property renaming.
|
121 | // The code must remain in the `result['domEventName']` form.
|
122 | // return {domEventName, fullKey};
|
123 | const result = {};
|
124 | result['domEventName'] = domEventName;
|
125 | result['fullKey'] = fullKey;
|
126 | return result;
|
127 | }
|
128 | static getEventFullKey(event) {
|
129 | let fullKey = '';
|
130 | let key = getEventKey(event);
|
131 | key = key.toLowerCase();
|
132 | if (key === ' ') {
|
133 | key = 'space'; // for readability
|
134 | }
|
135 | else if (key === '.') {
|
136 | key = 'dot'; // because '.' is used as a separator in event names
|
137 | }
|
138 | MODIFIER_KEYS.forEach(modifierName => {
|
139 | if (modifierName != key) {
|
140 | const modifierGetter = MODIFIER_KEY_GETTERS[modifierName];
|
141 | if (modifierGetter(event)) {
|
142 | fullKey += modifierName + '.';
|
143 | }
|
144 | }
|
145 | });
|
146 | fullKey += key;
|
147 | return fullKey;
|
148 | }
|
149 | /**
|
150 | * Configures a handler callback for a key event.
|
151 | * @param fullKey The event name that combines all simultaneous keystrokes.
|
152 | * @param handler The function that responds to the key event.
|
153 | * @param zone The zone in which the event occurred.
|
154 | * @returns A callback function.
|
155 | */
|
156 | static eventCallback(fullKey, handler, zone) {
|
157 | return (event /** TODO #9100 */) => {
|
158 | if (KeyEventsPlugin.getEventFullKey(event) === fullKey) {
|
159 | zone.runGuarded(() => handler(event));
|
160 | }
|
161 | };
|
162 | }
|
163 | /** @internal */
|
164 | static _normalizeKey(keyName) {
|
165 | // TODO: switch to a Map if the mapping grows too much
|
166 | switch (keyName) {
|
167 | case 'esc':
|
168 | return 'escape';
|
169 | default:
|
170 | return keyName;
|
171 | }
|
172 | }
|
173 | }
|
174 | KeyEventsPlugin.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.0.3", ngImport: i0, type: KeyEventsPlugin, deps: [{ token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Injectable });
|
175 | KeyEventsPlugin.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "13.0.3", ngImport: i0, type: KeyEventsPlugin });
|
176 | i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.0.3", ngImport: i0, type: KeyEventsPlugin, decorators: [{
|
177 | type: Injectable
|
178 | }], ctorParameters: function () { return [{ type: undefined, decorators: [{
|
179 | type: Inject,
|
180 | args: [DOCUMENT]
|
181 | }] }]; } });
|
182 | function getEventKey(event) {
|
183 | let key = event.key;
|
184 | if (key == null) {
|
185 | key = event.keyIdentifier;
|
186 | // keyIdentifier is defined in the old draft of DOM Level 3 Events implemented by Chrome and
|
187 | // Safari cf
|
188 | // https://www.w3.org/TR/2007/WD-DOM-Level-3-Events-20071221/events.html#Events-KeyboardEvents-Interfaces
|
189 | if (key == null) {
|
190 | return 'Unidentified';
|
191 | }
|
192 | if (key.startsWith('U+')) {
|
193 | key = String.fromCharCode(parseInt(key.substring(2), 16));
|
194 | if (event.location === DOM_KEY_LOCATION_NUMPAD && _chromeNumKeyPadMap.hasOwnProperty(key)) {
|
195 | // There is a bug in Chrome for numeric keypad keys:
|
196 | // https://code.google.com/p/chromium/issues/detail?id=155654
|
197 | // 1, 2, 3 ... are reported as A, B, C ...
|
198 | key = _chromeNumKeyPadMap[key];
|
199 | }
|
200 | }
|
201 | }
|
202 | return _keyMap[key] || key;
|
203 | }
|
204 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"key_events.js","sourceRoot":"","sources":["../../../../../../../../packages/platform-browser/src/dom/events/key_events.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {DOCUMENT, ɵgetDOM as getDOM} from '@angular/common';\nimport {Inject, Injectable, NgZone} from '@angular/core';\nimport {EventManagerPlugin} from './event_manager';\n\n/**\n * Defines supported modifiers for key events.\n */\nconst MODIFIER_KEYS = ['alt', 'control', 'meta', 'shift'];\n\nconst DOM_KEY_LOCATION_NUMPAD = 3;\n\n// Map to convert some key or keyIdentifier values to what will be returned by getEventKey\nconst _keyMap: {[k: string]: string} = {\n  // The following values are here for cross-browser compatibility and to match the W3C standard\n  // cf https://www.w3.org/TR/DOM-Level-3-Events-key/\n  '\\b': 'Backspace',\n  '\\t': 'Tab',\n  '\\x7F': 'Delete',\n  '\\x1B': 'Escape',\n  'Del': 'Delete',\n  'Esc': 'Escape',\n  'Left': 'ArrowLeft',\n  'Right': 'ArrowRight',\n  'Up': 'ArrowUp',\n  'Down': 'ArrowDown',\n  'Menu': 'ContextMenu',\n  'Scroll': 'ScrollLock',\n  'Win': 'OS'\n};\n\n// There is a bug in Chrome for numeric keypad keys:\n// https://code.google.com/p/chromium/issues/detail?id=155654\n// 1, 2, 3 ... are reported as A, B, C ...\nconst _chromeNumKeyPadMap = {\n  'A': '1',\n  'B': '2',\n  'C': '3',\n  'D': '4',\n  'E': '5',\n  'F': '6',\n  'G': '7',\n  'H': '8',\n  'I': '9',\n  'J': '*',\n  'K': '+',\n  'M': '-',\n  'N': '.',\n  'O': '/',\n  '\\x60': '0',\n  '\\x90': 'NumLock'\n};\n\n\n/**\n * Retrieves modifiers from key-event objects.\n */\nconst MODIFIER_KEY_GETTERS: {[key: string]: (event: KeyboardEvent) => boolean} = {\n  'alt': (event: KeyboardEvent) => event.altKey,\n  'control': (event: KeyboardEvent) => event.ctrlKey,\n  'meta': (event: KeyboardEvent) => event.metaKey,\n  'shift': (event: KeyboardEvent) => event.shiftKey\n};\n\n/**\n * @publicApi\n * A browser plug-in that provides support for handling of key events in Angular.\n */\n@Injectable()\nexport class KeyEventsPlugin extends EventManagerPlugin {\n  /**\n   * Initializes an instance of the browser plug-in.\n   * @param doc The document in which key events will be detected.\n   */\n  constructor(@Inject(DOCUMENT) doc: any) {\n    super(doc);\n  }\n\n  /**\n   * Reports whether a named key event is supported.\n   * @param eventName The event name to query.\n   * @return True if the named key event is supported.\n   */\n  override supports(eventName: string): boolean {\n    return KeyEventsPlugin.parseEventName(eventName) != null;\n  }\n\n  /**\n   * Registers a handler for a specific element and key event.\n   * @param element The HTML element to receive event notifications.\n   * @param eventName The name of the key event to listen for.\n   * @param handler A function to call when the notification occurs. Receives the\n   * event object as an argument.\n   * @returns The key event that was registered.\n   */\n  override addEventListener(element: HTMLElement, eventName: string, handler: Function): Function {\n    const parsedEvent = KeyEventsPlugin.parseEventName(eventName)!;\n\n    const outsideHandler =\n        KeyEventsPlugin.eventCallback(parsedEvent['fullKey'], handler, this.manager.getZone());\n\n    return this.manager.getZone().runOutsideAngular(() => {\n      return getDOM().onAndCancel(element, parsedEvent['domEventName'], outsideHandler);\n    });\n  }\n\n  static parseEventName(eventName: string): {fullKey: string, domEventName: string}|null {\n    const parts: string[] = eventName.toLowerCase().split('.');\n\n    const domEventName = parts.shift();\n    if ((parts.length === 0) || !(domEventName === 'keydown' || domEventName === 'keyup')) {\n      return null;\n    }\n\n    const key = KeyEventsPlugin._normalizeKey(parts.pop()!);\n\n    let fullKey = '';\n    MODIFIER_KEYS.forEach(modifierName => {\n      const index: number = parts.indexOf(modifierName);\n      if (index > -1) {\n        parts.splice(index, 1);\n        fullKey += modifierName + '.';\n      }\n    });\n    fullKey += key;\n\n    if (parts.length != 0 || key.length === 0) {\n      // returning null instead of throwing to let another plugin process the event\n      return null;\n    }\n\n    // NOTE: Please don't rewrite this as so, as it will break JSCompiler property renaming.\n    //       The code must remain in the `result['domEventName']` form.\n    // return {domEventName, fullKey};\n    const result: {fullKey: string, domEventName: string} = {} as any;\n    result['domEventName'] = domEventName;\n    result['fullKey'] = fullKey;\n    return result;\n  }\n\n  static getEventFullKey(event: KeyboardEvent): string {\n    let fullKey = '';\n    let key = getEventKey(event);\n    key = key.toLowerCase();\n    if (key === ' ') {\n      key = 'space';  // for readability\n    } else if (key === '.') {\n      key = 'dot';  // because '.' is used as a separator in event names\n    }\n    MODIFIER_KEYS.forEach(modifierName => {\n      if (modifierName != key) {\n        const modifierGetter = MODIFIER_KEY_GETTERS[modifierName];\n        if (modifierGetter(event)) {\n          fullKey += modifierName + '.';\n        }\n      }\n    });\n    fullKey += key;\n    return fullKey;\n  }\n\n  /**\n   * Configures a handler callback for a key event.\n   * @param fullKey The event name that combines all simultaneous keystrokes.\n   * @param handler The function that responds to the key event.\n   * @param zone The zone in which the event occurred.\n   * @returns A callback function.\n   */\n  static eventCallback(fullKey: any, handler: Function, zone: NgZone): Function {\n    return (event: any /** TODO #9100 */) => {\n      if (KeyEventsPlugin.getEventFullKey(event) === fullKey) {\n        zone.runGuarded(() => handler(event));\n      }\n    };\n  }\n\n  /** @internal */\n  static _normalizeKey(keyName: string): string {\n    // TODO: switch to a Map if the mapping grows too much\n    switch (keyName) {\n      case 'esc':\n        return 'escape';\n      default:\n        return keyName;\n    }\n  }\n}\n\nfunction getEventKey(event: any): string {\n  let key = event.key;\n  if (key == null) {\n    key = event.keyIdentifier;\n    // keyIdentifier is defined in the old draft of DOM Level 3 Events implemented by Chrome and\n    // Safari cf\n    // https://www.w3.org/TR/2007/WD-DOM-Level-3-Events-20071221/events.html#Events-KeyboardEvents-Interfaces\n    if (key == null) {\n      return 'Unidentified';\n    }\n    if (key.startsWith('U+')) {\n      key = String.fromCharCode(parseInt(key.substring(2), 16));\n      if (event.location === DOM_KEY_LOCATION_NUMPAD && _chromeNumKeyPadMap.hasOwnProperty(key)) {\n        // There is a bug in Chrome for numeric keypad keys:\n        // https://code.google.com/p/chromium/issues/detail?id=155654\n        // 1, 2, 3 ... are reported as A, B, C ...\n        key = (_chromeNumKeyPadMap as any)[key];\n      }\n    }\n  }\n\n  return _keyMap[key] || key;\n}\n"]} |
\ | No newline at end of file |