source: trip-planner-front/node_modules/@angular/cdk/esm2015/a11y/focus-monitor/focus-monitor.js@ e29cc2e

Last change on this file since e29cc2e was e29cc2e, checked in by Ema <ema_spirova@…>, 3 years ago

primeNG components

  • Property mode set to 100644
File size: 63.1 KB
Line 
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 */
8import { Platform, normalizePassiveListenerOptions, _getShadowRoot, _getEventTarget, } from '@angular/cdk/platform';
9import { Directive, ElementRef, EventEmitter, Inject, Injectable, InjectionToken, NgZone, Optional, Output, } from '@angular/core';
10import { of as observableOf, Subject } from 'rxjs';
11import { takeUntil } from 'rxjs/operators';
12import { coerceElement } from '@angular/cdk/coercion';
13import { DOCUMENT } from '@angular/common';
14import { InputModalityDetector, TOUCH_BUFFER_MS, } from '../input-modality/input-modality-detector';
15import * as i0 from "@angular/core";
16import * as i1 from "@angular/cdk/platform";
17import * as i2 from "../input-modality/input-modality-detector";
18import * as i3 from "@angular/common";
19/** InjectionToken for FocusMonitorOptions. */
20export const FOCUS_MONITOR_DEFAULT_OPTIONS = new InjectionToken('cdk-focus-monitor-default-options');
21/**
22 * Event listener options that enable capturing and also
23 * mark the listener as passive if the browser supports it.
24 */
25const captureEventListenerOptions = normalizePassiveListenerOptions({
26 passive: true,
27 capture: true
28});
29/** Monitors mouse and keyboard events to determine the cause of focus events. */
30export class FocusMonitor {
31 constructor(_ngZone, _platform, _inputModalityDetector,
32 /** @breaking-change 11.0.0 make document required */
33 document, options) {
34 this._ngZone = _ngZone;
35 this._platform = _platform;
36 this._inputModalityDetector = _inputModalityDetector;
37 /** The focus origin that the next focus event is a result of. */
38 this._origin = null;
39 /** Whether the window has just been focused. */
40 this._windowFocused = false;
41 /**
42 * Whether the origin was determined via a touch interaction. Necessary as properly attributing
43 * focus events to touch interactions requires special logic.
44 */
45 this._originFromTouchInteraction = false;
46 /** Map of elements being monitored to their info. */
47 this._elementInfo = new Map();
48 /** The number of elements currently being monitored. */
49 this._monitoredElementCount = 0;
50 /**
51 * Keeps track of the root nodes to which we've currently bound a focus/blur handler,
52 * as well as the number of monitored elements that they contain. We have to treat focus/blur
53 * handlers differently from the rest of the events, because the browser won't emit events
54 * to the document when focus moves inside of a shadow root.
55 */
56 this._rootNodeFocusListenerCount = new Map();
57 /**
58 * Event listener for `focus` events on the window.
59 * Needs to be an arrow function in order to preserve the context when it gets bound.
60 */
61 this._windowFocusListener = () => {
62 // Make a note of when the window regains focus, so we can
63 // restore the origin info for the focused element.
64 this._windowFocused = true;
65 this._windowFocusTimeoutId = setTimeout(() => this._windowFocused = false);
66 };
67 /** Subject for stopping our InputModalityDetector subscription. */
68 this._stopInputModalityDetector = new Subject();
69 /**
70 * Event listener for `focus` and 'blur' events on the document.
71 * Needs to be an arrow function in order to preserve the context when it gets bound.
72 */
73 this._rootNodeFocusAndBlurListener = (event) => {
74 const target = _getEventTarget(event);
75 const handler = event.type === 'focus' ? this._onFocus : this._onBlur;
76 // We need to walk up the ancestor chain in order to support `checkChildren`.
77 for (let element = target; element; element = element.parentElement) {
78 handler.call(this, event, element);
79 }
80 };
81 this._document = document;
82 this._detectionMode = (options === null || options === void 0 ? void 0 : options.detectionMode) || 0 /* IMMEDIATE */;
83 }
84 monitor(element, checkChildren = false) {
85 const nativeElement = coerceElement(element);
86 // Do nothing if we're not on the browser platform or the passed in node isn't an element.
87 if (!this._platform.isBrowser || nativeElement.nodeType !== 1) {
88 return observableOf(null);
89 }
90 // If the element is inside the shadow DOM, we need to bind our focus/blur listeners to
91 // the shadow root, rather than the `document`, because the browser won't emit focus events
92 // to the `document`, if focus is moving within the same shadow root.
93 const rootNode = _getShadowRoot(nativeElement) || this._getDocument();
94 const cachedInfo = this._elementInfo.get(nativeElement);
95 // Check if we're already monitoring this element.
96 if (cachedInfo) {
97 if (checkChildren) {
98 // TODO(COMP-318): this can be problematic, because it'll turn all non-checkChildren
99 // observers into ones that behave as if `checkChildren` was turned on. We need a more
100 // robust solution.
101 cachedInfo.checkChildren = true;
102 }
103 return cachedInfo.subject;
104 }
105 // Create monitored element info.
106 const info = {
107 checkChildren: checkChildren,
108 subject: new Subject(),
109 rootNode
110 };
111 this._elementInfo.set(nativeElement, info);
112 this._registerGlobalListeners(info);
113 return info.subject;
114 }
115 stopMonitoring(element) {
116 const nativeElement = coerceElement(element);
117 const elementInfo = this._elementInfo.get(nativeElement);
118 if (elementInfo) {
119 elementInfo.subject.complete();
120 this._setClasses(nativeElement);
121 this._elementInfo.delete(nativeElement);
122 this._removeGlobalListeners(elementInfo);
123 }
124 }
125 focusVia(element, origin, options) {
126 const nativeElement = coerceElement(element);
127 const focusedElement = this._getDocument().activeElement;
128 // If the element is focused already, calling `focus` again won't trigger the event listener
129 // which means that the focus classes won't be updated. If that's the case, update the classes
130 // directly without waiting for an event.
131 if (nativeElement === focusedElement) {
132 this._getClosestElementsInfo(nativeElement)
133 .forEach(([currentElement, info]) => this._originChanged(currentElement, origin, info));
134 }
135 else {
136 this._setOrigin(origin);
137 // `focus` isn't available on the server
138 if (typeof nativeElement.focus === 'function') {
139 nativeElement.focus(options);
140 }
141 }
142 }
143 ngOnDestroy() {
144 this._elementInfo.forEach((_info, element) => this.stopMonitoring(element));
145 }
146 /** Access injected document if available or fallback to global document reference */
147 _getDocument() {
148 return this._document || document;
149 }
150 /** Use defaultView of injected document if available or fallback to global window reference */
151 _getWindow() {
152 const doc = this._getDocument();
153 return doc.defaultView || window;
154 }
155 _toggleClass(element, className, shouldSet) {
156 if (shouldSet) {
157 element.classList.add(className);
158 }
159 else {
160 element.classList.remove(className);
161 }
162 }
163 _getFocusOrigin(focusEventTarget) {
164 if (this._origin) {
165 // If the origin was realized via a touch interaction, we need to perform additional checks
166 // to determine whether the focus origin should be attributed to touch or program.
167 if (this._originFromTouchInteraction) {
168 return this._shouldBeAttributedToTouch(focusEventTarget) ? 'touch' : 'program';
169 }
170 else {
171 return this._origin;
172 }
173 }
174 // If the window has just regained focus, we can restore the most recent origin from before the
175 // window blurred. Otherwise, we've reached the point where we can't identify the source of the
176 // focus. This typically means one of two things happened:
177 //
178 // 1) The element was programmatically focused, or
179 // 2) The element was focused via screen reader navigation (which generally doesn't fire
180 // events).
181 //
182 // Because we can't distinguish between these two cases, we default to setting `program`.
183 return (this._windowFocused && this._lastFocusOrigin) ? this._lastFocusOrigin : 'program';
184 }
185 /**
186 * Returns whether the focus event should be attributed to touch. Recall that in IMMEDIATE mode, a
187 * touch origin isn't immediately reset at the next tick (see _setOrigin). This means that when we
188 * handle a focus event following a touch interaction, we need to determine whether (1) the focus
189 * event was directly caused by the touch interaction or (2) the focus event was caused by a
190 * subsequent programmatic focus call triggered by the touch interaction.
191 * @param focusEventTarget The target of the focus event under examination.
192 */
193 _shouldBeAttributedToTouch(focusEventTarget) {
194 // Please note that this check is not perfect. Consider the following edge case:
195 //
196 // <div #parent tabindex="0">
197 // <div #child tabindex="0" (click)="#parent.focus()"></div>
198 // </div>
199 //
200 // Suppose there is a FocusMonitor in IMMEDIATE mode attached to #parent. When the user touches
201 // #child, #parent is programmatically focused. This code will attribute the focus to touch
202 // instead of program. This is a relatively minor edge-case that can be worked around by using
203 // focusVia(parent, 'program') to focus #parent.
204 return (this._detectionMode === 1 /* EVENTUAL */) ||
205 !!(focusEventTarget === null || focusEventTarget === void 0 ? void 0 : focusEventTarget.contains(this._inputModalityDetector._mostRecentTarget));
206 }
207 /**
208 * Sets the focus classes on the element based on the given focus origin.
209 * @param element The element to update the classes on.
210 * @param origin The focus origin.
211 */
212 _setClasses(element, origin) {
213 this._toggleClass(element, 'cdk-focused', !!origin);
214 this._toggleClass(element, 'cdk-touch-focused', origin === 'touch');
215 this._toggleClass(element, 'cdk-keyboard-focused', origin === 'keyboard');
216 this._toggleClass(element, 'cdk-mouse-focused', origin === 'mouse');
217 this._toggleClass(element, 'cdk-program-focused', origin === 'program');
218 }
219 /**
220 * Updates the focus origin. If we're using immediate detection mode, we schedule an async
221 * function to clear the origin at the end of a timeout. The duration of the timeout depends on
222 * the origin being set.
223 * @param origin The origin to set.
224 * @param isFromInteraction Whether we are setting the origin from an interaction event.
225 */
226 _setOrigin(origin, isFromInteraction = false) {
227 this._ngZone.runOutsideAngular(() => {
228 this._origin = origin;
229 this._originFromTouchInteraction = (origin === 'touch') && isFromInteraction;
230 // If we're in IMMEDIATE mode, reset the origin at the next tick (or in `TOUCH_BUFFER_MS` ms
231 // for a touch event). We reset the origin at the next tick because Firefox focuses one tick
232 // after the interaction event. We wait `TOUCH_BUFFER_MS` ms before resetting the origin for
233 // a touch event because when a touch event is fired, the associated focus event isn't yet in
234 // the event queue. Before doing so, clear any pending timeouts.
235 if (this._detectionMode === 0 /* IMMEDIATE */) {
236 clearTimeout(this._originTimeoutId);
237 const ms = this._originFromTouchInteraction ? TOUCH_BUFFER_MS : 1;
238 this._originTimeoutId = setTimeout(() => this._origin = null, ms);
239 }
240 });
241 }
242 /**
243 * Handles focus events on a registered element.
244 * @param event The focus event.
245 * @param element The monitored element.
246 */
247 _onFocus(event, element) {
248 // NOTE(mmalerba): We currently set the classes based on the focus origin of the most recent
249 // focus event affecting the monitored element. If we want to use the origin of the first event
250 // instead we should check for the cdk-focused class here and return if the element already has
251 // it. (This only matters for elements that have includesChildren = true).
252 // If we are not counting child-element-focus as focused, make sure that the event target is the
253 // monitored element itself.
254 const elementInfo = this._elementInfo.get(element);
255 const focusEventTarget = _getEventTarget(event);
256 if (!elementInfo || (!elementInfo.checkChildren && element !== focusEventTarget)) {
257 return;
258 }
259 this._originChanged(element, this._getFocusOrigin(focusEventTarget), elementInfo);
260 }
261 /**
262 * Handles blur events on a registered element.
263 * @param event The blur event.
264 * @param element The monitored element.
265 */
266 _onBlur(event, element) {
267 // If we are counting child-element-focus as focused, make sure that we aren't just blurring in
268 // order to focus another child of the monitored element.
269 const elementInfo = this._elementInfo.get(element);
270 if (!elementInfo || (elementInfo.checkChildren && event.relatedTarget instanceof Node &&
271 element.contains(event.relatedTarget))) {
272 return;
273 }
274 this._setClasses(element);
275 this._emitOrigin(elementInfo.subject, null);
276 }
277 _emitOrigin(subject, origin) {
278 this._ngZone.run(() => subject.next(origin));
279 }
280 _registerGlobalListeners(elementInfo) {
281 if (!this._platform.isBrowser) {
282 return;
283 }
284 const rootNode = elementInfo.rootNode;
285 const rootNodeFocusListeners = this._rootNodeFocusListenerCount.get(rootNode) || 0;
286 if (!rootNodeFocusListeners) {
287 this._ngZone.runOutsideAngular(() => {
288 rootNode.addEventListener('focus', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
289 rootNode.addEventListener('blur', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
290 });
291 }
292 this._rootNodeFocusListenerCount.set(rootNode, rootNodeFocusListeners + 1);
293 // Register global listeners when first element is monitored.
294 if (++this._monitoredElementCount === 1) {
295 // Note: we listen to events in the capture phase so we
296 // can detect them even if the user stops propagation.
297 this._ngZone.runOutsideAngular(() => {
298 const window = this._getWindow();
299 window.addEventListener('focus', this._windowFocusListener);
300 });
301 // The InputModalityDetector is also just a collection of global listeners.
302 this._inputModalityDetector.modalityDetected
303 .pipe(takeUntil(this._stopInputModalityDetector))
304 .subscribe(modality => { this._setOrigin(modality, true /* isFromInteraction */); });
305 }
306 }
307 _removeGlobalListeners(elementInfo) {
308 const rootNode = elementInfo.rootNode;
309 if (this._rootNodeFocusListenerCount.has(rootNode)) {
310 const rootNodeFocusListeners = this._rootNodeFocusListenerCount.get(rootNode);
311 if (rootNodeFocusListeners > 1) {
312 this._rootNodeFocusListenerCount.set(rootNode, rootNodeFocusListeners - 1);
313 }
314 else {
315 rootNode.removeEventListener('focus', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
316 rootNode.removeEventListener('blur', this._rootNodeFocusAndBlurListener, captureEventListenerOptions);
317 this._rootNodeFocusListenerCount.delete(rootNode);
318 }
319 }
320 // Unregister global listeners when last element is unmonitored.
321 if (!--this._monitoredElementCount) {
322 const window = this._getWindow();
323 window.removeEventListener('focus', this._windowFocusListener);
324 // Equivalently, stop our InputModalityDetector subscription.
325 this._stopInputModalityDetector.next();
326 // Clear timeouts for all potentially pending timeouts to prevent the leaks.
327 clearTimeout(this._windowFocusTimeoutId);
328 clearTimeout(this._originTimeoutId);
329 }
330 }
331 /** Updates all the state on an element once its focus origin has changed. */
332 _originChanged(element, origin, elementInfo) {
333 this._setClasses(element, origin);
334 this._emitOrigin(elementInfo.subject, origin);
335 this._lastFocusOrigin = origin;
336 }
337 /**
338 * Collects the `MonitoredElementInfo` of a particular element and
339 * all of its ancestors that have enabled `checkChildren`.
340 * @param element Element from which to start the search.
341 */
342 _getClosestElementsInfo(element) {
343 const results = [];
344 this._elementInfo.forEach((info, currentElement) => {
345 if (currentElement === element || (info.checkChildren && currentElement.contains(element))) {
346 results.push([currentElement, info]);
347 }
348 });
349 return results;
350 }
351}
352FocusMonitor.ɵprov = i0.ɵɵdefineInjectable({ factory: function FocusMonitor_Factory() { return new FocusMonitor(i0.ɵɵinject(i0.NgZone), i0.ɵɵinject(i1.Platform), i0.ɵɵinject(i2.InputModalityDetector), i0.ɵɵinject(i3.DOCUMENT, 8), i0.ɵɵinject(FOCUS_MONITOR_DEFAULT_OPTIONS, 8)); }, token: FocusMonitor, providedIn: "root" });
353FocusMonitor.decorators = [
354 { type: Injectable, args: [{ providedIn: 'root' },] }
355];
356FocusMonitor.ctorParameters = () => [
357 { type: NgZone },
358 { type: Platform },
359 { type: InputModalityDetector },
360 { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [DOCUMENT,] }] },
361 { type: undefined, decorators: [{ type: Optional }, { type: Inject, args: [FOCUS_MONITOR_DEFAULT_OPTIONS,] }] }
362];
363/**
364 * Directive that determines how a particular element was focused (via keyboard, mouse, touch, or
365 * programmatically) and adds corresponding classes to the element.
366 *
367 * There are two variants of this directive:
368 * 1) cdkMonitorElementFocus: does not consider an element to be focused if one of its children is
369 * focused.
370 * 2) cdkMonitorSubtreeFocus: considers an element focused if it or any of its children are focused.
371 */
372export class CdkMonitorFocus {
373 constructor(_elementRef, _focusMonitor) {
374 this._elementRef = _elementRef;
375 this._focusMonitor = _focusMonitor;
376 this.cdkFocusChange = new EventEmitter();
377 }
378 ngAfterViewInit() {
379 const element = this._elementRef.nativeElement;
380 this._monitorSubscription = this._focusMonitor.monitor(element, element.nodeType === 1 && element.hasAttribute('cdkMonitorSubtreeFocus'))
381 .subscribe(origin => this.cdkFocusChange.emit(origin));
382 }
383 ngOnDestroy() {
384 this._focusMonitor.stopMonitoring(this._elementRef);
385 if (this._monitorSubscription) {
386 this._monitorSubscription.unsubscribe();
387 }
388 }
389}
390CdkMonitorFocus.decorators = [
391 { type: Directive, args: [{
392 selector: '[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]',
393 },] }
394];
395CdkMonitorFocus.ctorParameters = () => [
396 { type: ElementRef },
397 { type: FocusMonitor }
398];
399CdkMonitorFocus.propDecorators = {
400 cdkFocusChange: [{ type: Output }]
401};
402//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"focus-monitor.js","sourceRoot":"","sources":["../../../../../../../src/cdk/a11y/focus-monitor/focus-monitor.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 {\n  Platform,\n  normalizePassiveListenerOptions,\n  _getShadowRoot,\n  _getEventTarget,\n} from '@angular/cdk/platform';\nimport {\n  Directive,\n  ElementRef,\n  EventEmitter,\n  Inject,\n  Injectable,\n  InjectionToken,\n  NgZone,\n  OnDestroy,\n  Optional,\n  Output,\n  AfterViewInit,\n} from '@angular/core';\nimport {Observable, of as observableOf, Subject, Subscription} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\nimport {coerceElement} from '@angular/cdk/coercion';\nimport {DOCUMENT} from '@angular/common';\nimport {\n  InputModalityDetector,\n  TOUCH_BUFFER_MS,\n} from '../input-modality/input-modality-detector';\n\n\nexport type FocusOrigin = 'touch' | 'mouse' | 'keyboard' | 'program' | null;\n\n/**\n * Corresponds to the options that can be passed to the native `focus` event.\n * via https://developer.mozilla.org/en-US/docs/Web/API/HTMLElement/focus\n */\nexport interface FocusOptions {\n  /** Whether the browser should scroll to the element when it is focused. */\n  preventScroll?: boolean;\n}\n\n/** Detection mode used for attributing the origin of a focus event. */\nexport const enum FocusMonitorDetectionMode {\n  /**\n   * Any mousedown, keydown, or touchstart event that happened in the previous\n   * tick or the current tick will be used to assign a focus event's origin (to\n   * either mouse, keyboard, or touch). This is the default option.\n   */\n  IMMEDIATE,\n  /**\n   * A focus event's origin is always attributed to the last corresponding\n   * mousedown, keydown, or touchstart event, no matter how long ago it occurred.\n   */\n  EVENTUAL\n}\n\n/** Injectable service-level options for FocusMonitor. */\nexport interface FocusMonitorOptions {\n  detectionMode?: FocusMonitorDetectionMode;\n}\n\n/** InjectionToken for FocusMonitorOptions. */\nexport const FOCUS_MONITOR_DEFAULT_OPTIONS =\n    new InjectionToken<FocusMonitorOptions>('cdk-focus-monitor-default-options');\n\ntype MonitoredElementInfo = {\n  checkChildren: boolean,\n  readonly subject: Subject<FocusOrigin>,\n  rootNode: HTMLElement|ShadowRoot|Document\n};\n\n/**\n * Event listener options that enable capturing and also\n * mark the listener as passive if the browser supports it.\n */\nconst captureEventListenerOptions = normalizePassiveListenerOptions({\n  passive: true,\n  capture: true\n});\n\n\n/** Monitors mouse and keyboard events to determine the cause of focus events. */\n@Injectable({providedIn: 'root'})\nexport class FocusMonitor implements OnDestroy {\n  /** The focus origin that the next focus event is a result of. */\n  private _origin: FocusOrigin = null;\n\n  /** The FocusOrigin of the last focus event tracked by the FocusMonitor. */\n  private _lastFocusOrigin: FocusOrigin;\n\n  /** Whether the window has just been focused. */\n  private _windowFocused = false;\n\n  /** The timeout id of the window focus timeout. */\n  private _windowFocusTimeoutId: number;\n\n  /** The timeout id of the origin clearing timeout. */\n  private _originTimeoutId: number;\n\n  /**\n   * Whether the origin was determined via a touch interaction. Necessary as properly attributing\n   * focus events to touch interactions requires special logic.\n   */\n  private _originFromTouchInteraction = false;\n\n  /** Map of elements being monitored to their info. */\n  private _elementInfo = new Map<HTMLElement, MonitoredElementInfo>();\n\n  /** The number of elements currently being monitored. */\n  private _monitoredElementCount = 0;\n\n  /**\n   * Keeps track of the root nodes to which we've currently bound a focus/blur handler,\n   * as well as the number of monitored elements that they contain. We have to treat focus/blur\n   * handlers differently from the rest of the events, because the browser won't emit events\n   * to the document when focus moves inside of a shadow root.\n   */\n  private _rootNodeFocusListenerCount = new Map<HTMLElement|Document|ShadowRoot, number>();\n\n  /**\n   * The specified detection mode, used for attributing the origin of a focus\n   * event.\n   */\n  private readonly _detectionMode: FocusMonitorDetectionMode;\n\n  /**\n   * Event listener for `focus` events on the window.\n   * Needs to be an arrow function in order to preserve the context when it gets bound.\n   */\n  private _windowFocusListener = () => {\n    // Make a note of when the window regains focus, so we can\n    // restore the origin info for the focused element.\n    this._windowFocused = true;\n    this._windowFocusTimeoutId = setTimeout(() => this._windowFocused = false);\n  }\n\n  /** Used to reference correct document/window */\n  protected _document?: Document;\n\n  /** Subject for stopping our InputModalityDetector subscription. */\n  private readonly _stopInputModalityDetector = new Subject<void>();\n\n  constructor(\n      private _ngZone: NgZone,\n      private _platform: Platform,\n      private readonly _inputModalityDetector: InputModalityDetector,\n      /** @breaking-change 11.0.0 make document required */\n      @Optional() @Inject(DOCUMENT) document: any|null,\n      @Optional() @Inject(FOCUS_MONITOR_DEFAULT_OPTIONS) options:\n          FocusMonitorOptions|null) {\n    this._document = document;\n    this._detectionMode = options?.detectionMode || FocusMonitorDetectionMode.IMMEDIATE;\n  }\n  /**\n   * Event listener for `focus` and 'blur' events on the document.\n   * Needs to be an arrow function in order to preserve the context when it gets bound.\n   */\n  private _rootNodeFocusAndBlurListener = (event: Event) => {\n    const target = _getEventTarget<HTMLElement>(event);\n    const handler = event.type === 'focus' ? this._onFocus : this._onBlur;\n\n    // We need to walk up the ancestor chain in order to support `checkChildren`.\n    for (let element = target; element; element = element.parentElement) {\n      handler.call(this, event as FocusEvent, element);\n    }\n  }\n\n  /**\n   * Monitors focus on an element and applies appropriate CSS classes.\n   * @param element The element to monitor\n   * @param checkChildren Whether to count the element as focused when its children are focused.\n   * @returns An observable that emits when the focus state of the element changes.\n   *     When the element is blurred, null will be emitted.\n   */\n  monitor(element: HTMLElement, checkChildren?: boolean): Observable<FocusOrigin>;\n\n  /**\n   * Monitors focus on an element and applies appropriate CSS classes.\n   * @param element The element to monitor\n   * @param checkChildren Whether to count the element as focused when its children are focused.\n   * @returns An observable that emits when the focus state of the element changes.\n   *     When the element is blurred, null will be emitted.\n   */\n  monitor(element: ElementRef<HTMLElement>, checkChildren?: boolean): Observable<FocusOrigin>;\n\n  monitor(element: HTMLElement | ElementRef<HTMLElement>,\n          checkChildren: boolean = false): Observable<FocusOrigin> {\n    const nativeElement = coerceElement(element);\n\n    // Do nothing if we're not on the browser platform or the passed in node isn't an element.\n    if (!this._platform.isBrowser || nativeElement.nodeType !== 1) {\n      return observableOf(null);\n    }\n\n    // If the element is inside the shadow DOM, we need to bind our focus/blur listeners to\n    // the shadow root, rather than the `document`, because the browser won't emit focus events\n    // to the `document`, if focus is moving within the same shadow root.\n    const rootNode = _getShadowRoot(nativeElement) || this._getDocument();\n    const cachedInfo = this._elementInfo.get(nativeElement);\n\n    // Check if we're already monitoring this element.\n    if (cachedInfo) {\n      if (checkChildren) {\n        // TODO(COMP-318): this can be problematic, because it'll turn all non-checkChildren\n        // observers into ones that behave as if `checkChildren` was turned on. We need a more\n        // robust solution.\n        cachedInfo.checkChildren = true;\n      }\n\n      return cachedInfo.subject;\n    }\n\n    // Create monitored element info.\n    const info: MonitoredElementInfo = {\n      checkChildren: checkChildren,\n      subject: new Subject<FocusOrigin>(),\n      rootNode\n    };\n    this._elementInfo.set(nativeElement, info);\n    this._registerGlobalListeners(info);\n\n    return info.subject;\n  }\n\n  /**\n   * Stops monitoring an element and removes all focus classes.\n   * @param element The element to stop monitoring.\n   */\n  stopMonitoring(element: HTMLElement): void;\n\n  /**\n   * Stops monitoring an element and removes all focus classes.\n   * @param element The element to stop monitoring.\n   */\n  stopMonitoring(element: ElementRef<HTMLElement>): void;\n\n  stopMonitoring(element: HTMLElement | ElementRef<HTMLElement>): void {\n    const nativeElement = coerceElement(element);\n    const elementInfo = this._elementInfo.get(nativeElement);\n\n    if (elementInfo) {\n      elementInfo.subject.complete();\n\n      this._setClasses(nativeElement);\n      this._elementInfo.delete(nativeElement);\n      this._removeGlobalListeners(elementInfo);\n    }\n  }\n\n  /**\n   * Focuses the element via the specified focus origin.\n   * @param element Element to focus.\n   * @param origin Focus origin.\n   * @param options Options that can be used to configure the focus behavior.\n   */\n  focusVia(element: HTMLElement, origin: FocusOrigin, options?: FocusOptions): void;\n\n  /**\n   * Focuses the element via the specified focus origin.\n   * @param element Element to focus.\n   * @param origin Focus origin.\n   * @param options Options that can be used to configure the focus behavior.\n   */\n  focusVia(element: ElementRef<HTMLElement>, origin: FocusOrigin, options?: FocusOptions): void;\n\n  focusVia(element: HTMLElement | ElementRef<HTMLElement>,\n          origin: FocusOrigin,\n          options?: FocusOptions): void {\n\n    const nativeElement = coerceElement(element);\n    const focusedElement = this._getDocument().activeElement;\n\n    // If the element is focused already, calling `focus` again won't trigger the event listener\n    // which means that the focus classes won't be updated. If that's the case, update the classes\n    // directly without waiting for an event.\n    if (nativeElement === focusedElement) {\n      this._getClosestElementsInfo(nativeElement)\n        .forEach(([currentElement, info]) => this._originChanged(currentElement, origin, info));\n    } else {\n      this._setOrigin(origin);\n\n      // `focus` isn't available on the server\n      if (typeof nativeElement.focus === 'function') {\n        nativeElement.focus(options);\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    this._elementInfo.forEach((_info, element) => this.stopMonitoring(element));\n  }\n\n  /** Access injected document if available or fallback to global document reference */\n  private _getDocument(): Document {\n    return this._document || document;\n  }\n\n  /** Use defaultView of injected document if available or fallback to global window reference */\n  private _getWindow(): Window {\n    const doc = this._getDocument();\n    return doc.defaultView || window;\n  }\n\n  private _toggleClass(element: Element, className: string, shouldSet: boolean) {\n    if (shouldSet) {\n      element.classList.add(className);\n    } else {\n      element.classList.remove(className);\n    }\n  }\n\n  private _getFocusOrigin(focusEventTarget: HTMLElement | null): FocusOrigin {\n    if (this._origin) {\n      // If the origin was realized via a touch interaction, we need to perform additional checks\n      // to determine whether the focus origin should be attributed to touch or program.\n      if (this._originFromTouchInteraction) {\n        return this._shouldBeAttributedToTouch(focusEventTarget) ? 'touch' : 'program';\n      } else {\n        return this._origin;\n      }\n    }\n\n    // If the window has just regained focus, we can restore the most recent origin from before the\n    // window blurred. Otherwise, we've reached the point where we can't identify the source of the\n    // focus. This typically means one of two things happened:\n    //\n    // 1) The element was programmatically focused, or\n    // 2) The element was focused via screen reader navigation (which generally doesn't fire\n    //    events).\n    //\n    // Because we can't distinguish between these two cases, we default to setting `program`.\n    return (this._windowFocused && this._lastFocusOrigin) ? this._lastFocusOrigin : 'program';\n  }\n\n  /**\n   * Returns whether the focus event should be attributed to touch. Recall that in IMMEDIATE mode, a\n   * touch origin isn't immediately reset at the next tick (see _setOrigin). This means that when we\n   * handle a focus event following a touch interaction, we need to determine whether (1) the focus\n   * event was directly caused by the touch interaction or (2) the focus event was caused by a\n   * subsequent programmatic focus call triggered by the touch interaction.\n   * @param focusEventTarget The target of the focus event under examination.\n   */\n  private _shouldBeAttributedToTouch(focusEventTarget: HTMLElement | null): boolean {\n    // Please note that this check is not perfect. Consider the following edge case:\n    //\n    // <div #parent tabindex=\"0\">\n    //   <div #child tabindex=\"0\" (click)=\"#parent.focus()\"></div>\n    // </div>\n    //\n    // Suppose there is a FocusMonitor in IMMEDIATE mode attached to #parent. When the user touches\n    // #child, #parent is programmatically focused. This code will attribute the focus to touch\n    // instead of program. This is a relatively minor edge-case that can be worked around by using\n    // focusVia(parent, 'program') to focus #parent.\n    return (this._detectionMode === FocusMonitorDetectionMode.EVENTUAL) ||\n        !!focusEventTarget?.contains(this._inputModalityDetector._mostRecentTarget);\n  }\n\n  /**\n   * Sets the focus classes on the element based on the given focus origin.\n   * @param element The element to update the classes on.\n   * @param origin The focus origin.\n   */\n  private _setClasses(element: HTMLElement, origin?: FocusOrigin): void {\n    this._toggleClass(element, 'cdk-focused', !!origin);\n    this._toggleClass(element, 'cdk-touch-focused', origin === 'touch');\n    this._toggleClass(element, 'cdk-keyboard-focused', origin === 'keyboard');\n    this._toggleClass(element, 'cdk-mouse-focused', origin === 'mouse');\n    this._toggleClass(element, 'cdk-program-focused', origin === 'program');\n  }\n\n  /**\n   * Updates the focus origin. If we're using immediate detection mode, we schedule an async\n   * function to clear the origin at the end of a timeout. The duration of the timeout depends on\n   * the origin being set.\n   * @param origin The origin to set.\n   * @param isFromInteraction Whether we are setting the origin from an interaction event.\n   */\n  private _setOrigin(origin: FocusOrigin, isFromInteraction = false): void {\n    this._ngZone.runOutsideAngular(() => {\n      this._origin = origin;\n      this._originFromTouchInteraction = (origin === 'touch') && isFromInteraction;\n\n      // If we're in IMMEDIATE mode, reset the origin at the next tick (or in `TOUCH_BUFFER_MS` ms\n      // for a touch event). We reset the origin at the next tick because Firefox focuses one tick\n      // after the interaction event. We wait `TOUCH_BUFFER_MS` ms before resetting the origin for\n      // a touch event because when a touch event is fired, the associated focus event isn't yet in\n      // the event queue. Before doing so, clear any pending timeouts.\n      if (this._detectionMode === FocusMonitorDetectionMode.IMMEDIATE) {\n        clearTimeout(this._originTimeoutId);\n        const ms = this._originFromTouchInteraction ? TOUCH_BUFFER_MS : 1;\n        this._originTimeoutId = setTimeout(() => this._origin = null, ms);\n      }\n    });\n  }\n\n  /**\n   * Handles focus events on a registered element.\n   * @param event The focus event.\n   * @param element The monitored element.\n   */\n  private _onFocus(event: FocusEvent, element: HTMLElement) {\n    // NOTE(mmalerba): We currently set the classes based on the focus origin of the most recent\n    // focus event affecting the monitored element. If we want to use the origin of the first event\n    // instead we should check for the cdk-focused class here and return if the element already has\n    // it. (This only matters for elements that have includesChildren = true).\n\n    // If we are not counting child-element-focus as focused, make sure that the event target is the\n    // monitored element itself.\n    const elementInfo = this._elementInfo.get(element);\n    const focusEventTarget = _getEventTarget<HTMLElement>(event);\n    if (!elementInfo || (!elementInfo.checkChildren && element !== focusEventTarget)) {\n      return;\n    }\n\n    this._originChanged(element, this._getFocusOrigin(focusEventTarget), elementInfo);\n  }\n\n  /**\n   * Handles blur events on a registered element.\n   * @param event The blur event.\n   * @param element The monitored element.\n   */\n  _onBlur(event: FocusEvent, element: HTMLElement) {\n    // If we are counting child-element-focus as focused, make sure that we aren't just blurring in\n    // order to focus another child of the monitored element.\n    const elementInfo = this._elementInfo.get(element);\n\n    if (!elementInfo || (elementInfo.checkChildren && event.relatedTarget instanceof Node &&\n        element.contains(event.relatedTarget))) {\n      return;\n    }\n\n    this._setClasses(element);\n    this._emitOrigin(elementInfo.subject, null);\n  }\n\n  private _emitOrigin(subject: Subject<FocusOrigin>, origin: FocusOrigin) {\n    this._ngZone.run(() => subject.next(origin));\n  }\n\n  private _registerGlobalListeners(elementInfo: MonitoredElementInfo) {\n    if (!this._platform.isBrowser) {\n      return;\n    }\n\n    const rootNode = elementInfo.rootNode;\n    const rootNodeFocusListeners = this._rootNodeFocusListenerCount.get(rootNode) || 0;\n\n    if (!rootNodeFocusListeners) {\n      this._ngZone.runOutsideAngular(() => {\n        rootNode.addEventListener('focus', this._rootNodeFocusAndBlurListener,\n          captureEventListenerOptions);\n        rootNode.addEventListener('blur', this._rootNodeFocusAndBlurListener,\n          captureEventListenerOptions);\n      });\n    }\n\n    this._rootNodeFocusListenerCount.set(rootNode, rootNodeFocusListeners + 1);\n\n    // Register global listeners when first element is monitored.\n    if (++this._monitoredElementCount === 1) {\n      // Note: we listen to events in the capture phase so we\n      // can detect them even if the user stops propagation.\n      this._ngZone.runOutsideAngular(() => {\n        const window = this._getWindow();\n        window.addEventListener('focus', this._windowFocusListener);\n      });\n\n      // The InputModalityDetector is also just a collection of global listeners.\n      this._inputModalityDetector.modalityDetected\n        .pipe(takeUntil(this._stopInputModalityDetector))\n        .subscribe(modality => { this._setOrigin(modality, true /* isFromInteraction */); });\n    }\n  }\n\n  private _removeGlobalListeners(elementInfo: MonitoredElementInfo) {\n    const rootNode = elementInfo.rootNode;\n\n    if (this._rootNodeFocusListenerCount.has(rootNode)) {\n      const rootNodeFocusListeners = this._rootNodeFocusListenerCount.get(rootNode)!;\n\n      if (rootNodeFocusListeners > 1) {\n        this._rootNodeFocusListenerCount.set(rootNode, rootNodeFocusListeners - 1);\n      } else {\n        rootNode.removeEventListener('focus', this._rootNodeFocusAndBlurListener,\n          captureEventListenerOptions);\n        rootNode.removeEventListener('blur', this._rootNodeFocusAndBlurListener,\n          captureEventListenerOptions);\n        this._rootNodeFocusListenerCount.delete(rootNode);\n      }\n    }\n\n    // Unregister global listeners when last element is unmonitored.\n    if (!--this._monitoredElementCount) {\n      const window = this._getWindow();\n      window.removeEventListener('focus', this._windowFocusListener);\n\n      // Equivalently, stop our InputModalityDetector subscription.\n      this._stopInputModalityDetector.next();\n\n      // Clear timeouts for all potentially pending timeouts to prevent the leaks.\n      clearTimeout(this._windowFocusTimeoutId);\n      clearTimeout(this._originTimeoutId);\n    }\n  }\n\n  /** Updates all the state on an element once its focus origin has changed. */\n  private _originChanged(element: HTMLElement, origin: FocusOrigin,\n                         elementInfo: MonitoredElementInfo) {\n    this._setClasses(element, origin);\n    this._emitOrigin(elementInfo.subject, origin);\n    this._lastFocusOrigin = origin;\n  }\n\n  /**\n   * Collects the `MonitoredElementInfo` of a particular element and\n   * all of its ancestors that have enabled `checkChildren`.\n   * @param element Element from which to start the search.\n   */\n  private _getClosestElementsInfo(element: HTMLElement): [HTMLElement, MonitoredElementInfo][] {\n    const results: [HTMLElement, MonitoredElementInfo][] = [];\n\n    this._elementInfo.forEach((info, currentElement) => {\n      if (currentElement === element || (info.checkChildren && currentElement.contains(element))) {\n        results.push([currentElement, info]);\n      }\n    });\n\n    return results;\n  }\n}\n\n/**\n * Directive that determines how a particular element was focused (via keyboard, mouse, touch, or\n * programmatically) and adds corresponding classes to the element.\n *\n * There are two variants of this directive:\n * 1) cdkMonitorElementFocus: does not consider an element to be focused if one of its children is\n *    focused.\n * 2) cdkMonitorSubtreeFocus: considers an element focused if it or any of its children are focused.\n */\n@Directive({\n  selector: '[cdkMonitorElementFocus], [cdkMonitorSubtreeFocus]',\n})\nexport class CdkMonitorFocus implements AfterViewInit, OnDestroy {\n  private _monitorSubscription: Subscription;\n  @Output() readonly cdkFocusChange = new EventEmitter<FocusOrigin>();\n\n  constructor(private _elementRef: ElementRef<HTMLElement>, private _focusMonitor: FocusMonitor) {}\n\n  ngAfterViewInit() {\n    const element = this._elementRef.nativeElement;\n    this._monitorSubscription = this._focusMonitor.monitor(\n      element,\n      element.nodeType === 1 && element.hasAttribute('cdkMonitorSubtreeFocus'))\n    .subscribe(origin => this.cdkFocusChange.emit(origin));\n  }\n\n  ngOnDestroy() {\n    this._focusMonitor.stopMonitoring(this._elementRef);\n\n    if (this._monitorSubscription) {\n      this._monitorSubscription.unsubscribe();\n    }\n  }\n}\n"]}
Note: See TracBrowser for help on using the repository browser.