@react-aria/landmark 3.0.0-alpha.4 → 3.0.0-alpha.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/main.js CHANGED
@@ -1,69 +1,114 @@
1
+ var $8Ore6$swchelperslib_define_propertyjs = require("@swc/helpers/lib/_define_property.js");
1
2
  var $8Ore6$react = require("react");
2
3
  var $8Ore6$reactariautils = require("@react-aria/utils");
4
+ var $8Ore6$usesyncexternalstoreshim = require("use-sync-external-store/shim");
3
5
 
4
6
  function $parcel$export(e, n, v, s) {
5
7
  Object.defineProperty(e, n, {get: v, set: s, enumerable: true, configurable: true});
6
8
  }
9
+ function $parcel$interopDefault(a) {
10
+ return a && a.__esModule ? a.default : a;
11
+ }
7
12
 
8
13
  $parcel$export(module.exports, "useLandmark", () => $202c109aedff6705$export$4cc632584fd87fae);
14
+ $parcel$export(module.exports, "createLandmarkController", () => $202c109aedff6705$export$f50151dbd51cd1d9);
15
+ /*
16
+ * Copyright 2022 Adobe. All rights reserved.
17
+ * This file is licensed to you under the Apache License, Version 2.0 (the "License");
18
+ * you may not use this file except in compliance with the License. You may obtain a copy
19
+ * of the License at http://www.apache.org/licenses/LICENSE-2.0
20
+ *
21
+ * Unless required by applicable law or agreed to in writing, software distributed under
22
+ * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
23
+ * OF ANY KIND, either express or implied. See the License for the specific language
24
+ * governing permissions and limitations under the License.
25
+ */ /*
26
+ * Copyright 2022 Adobe. All rights reserved.
27
+ * This file is licensed to you under the Apache License, Version 2.0 (the "License");
28
+ * you may not use this file except in compliance with the License. You may obtain a copy
29
+ * of the License at http://www.apache.org/licenses/LICENSE-2.0
30
+ *
31
+ * Unless required by applicable law or agreed to in writing, software distributed under
32
+ * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
33
+ * OF ANY KIND, either express or implied. See the License for the specific language
34
+ * governing permissions and limitations under the License.
35
+ */
36
+
9
37
 
10
38
 
39
+ // Increment this version number whenever the
40
+ // LandmarkManagerApi or Landmark interfaces change.
41
+ const $202c109aedff6705$var$LANDMARK_API_VERSION = 1;
42
+ // Symbol under which the singleton landmark manager instance is attached to the document.
43
+ const $202c109aedff6705$var$landmarkSymbol = Symbol.for("react-aria-landmark-manager");
44
+ function $202c109aedff6705$var$subscribe(fn) {
45
+ document.addEventListener("react-aria-landmark-manager-change", fn);
46
+ return ()=>document.removeEventListener("react-aria-landmark-manager-change", fn);
47
+ }
48
+ function $202c109aedff6705$var$getLandmarkManager() {
49
+ // Reuse an existing instance if it has the same or greater version.
50
+ let instance = document[$202c109aedff6705$var$landmarkSymbol];
51
+ if (instance && instance.version >= $202c109aedff6705$var$LANDMARK_API_VERSION) return instance;
52
+ // Otherwise, create a new instance and dispatch an event so anything using the existing
53
+ // instance updates and re-registers their landmarks with the new one.
54
+ document[$202c109aedff6705$var$landmarkSymbol] = new $202c109aedff6705$var$LandmarkManager();
55
+ document.dispatchEvent(new CustomEvent("react-aria-landmark-manager-change"));
56
+ return document[$202c109aedff6705$var$landmarkSymbol];
57
+ }
58
+ // Subscribes a React component to the current landmark manager instance.
59
+ function $202c109aedff6705$var$useLandmarkManager() {
60
+ return (0, $8Ore6$usesyncexternalstoreshim.useSyncExternalStore)($202c109aedff6705$var$subscribe, $202c109aedff6705$var$getLandmarkManager);
61
+ }
11
62
  class $202c109aedff6705$var$LandmarkManager {
12
- static getInstance() {
13
- if (!$202c109aedff6705$var$LandmarkManager.instance) $202c109aedff6705$var$LandmarkManager.instance = new $202c109aedff6705$var$LandmarkManager();
14
- return $202c109aedff6705$var$LandmarkManager.instance;
15
- }
16
- setup() {
17
- document.addEventListener('keydown', this.f6Handler, {
63
+ setupIfNeeded() {
64
+ if (this.isListening) return;
65
+ document.addEventListener("keydown", this.f6Handler, {
18
66
  capture: true
19
67
  });
20
- document.addEventListener('focusin', this.focusinHandler, {
68
+ document.addEventListener("focusin", this.focusinHandler, {
21
69
  capture: true
22
70
  });
23
- document.addEventListener('focusout', this.focusoutHandler, {
71
+ document.addEventListener("focusout", this.focusoutHandler, {
24
72
  capture: true
25
73
  });
26
74
  this.isListening = true;
27
75
  }
28
- teardown() {
29
- document.removeEventListener('keydown', this.f6Handler, {
76
+ teardownIfNeeded() {
77
+ if (!this.isListening || this.landmarks.length > 0 || this.refCount > 0) return;
78
+ document.removeEventListener("keydown", this.f6Handler, {
30
79
  capture: true
31
80
  });
32
- document.removeEventListener('focusin', this.focusinHandler, {
81
+ document.removeEventListener("focusin", this.focusinHandler, {
33
82
  capture: true
34
83
  });
35
- document.removeEventListener('focusout', this.focusoutHandler, {
84
+ document.removeEventListener("focusout", this.focusoutHandler, {
36
85
  capture: true
37
86
  });
38
87
  this.isListening = false;
39
88
  }
40
- focusLandmark(landmark) {
41
- var ref;
42
- (ref = this.landmarks.find((l)=>l.ref.current === landmark
43
- )) === null || ref === void 0 ? void 0 : ref.focus();
89
+ focusLandmark(landmark, direction) {
90
+ var _this_landmarks_find;
91
+ (_this_landmarks_find = this.landmarks.find((l)=>l.ref.current === landmark)) === null || _this_landmarks_find === void 0 ? void 0 : _this_landmarks_find.focus(direction);
44
92
  }
45
93
  /**
46
94
  * Return set of landmarks with a specific role.
47
95
  */ getLandmarksByRole(role) {
48
- return new Set(this.landmarks.filter((l)=>l.role === role
49
- ));
96
+ return new Set(this.landmarks.filter((l)=>l.role === role));
50
97
  }
51
98
  /**
52
99
  * Return first landmark with a specific role.
53
100
  */ getLandmarkByRole(role) {
54
- return this.landmarks.find((l)=>l.role === role
55
- );
101
+ return this.landmarks.find((l)=>l.role === role);
56
102
  }
57
103
  addLandmark(newLandmark) {
58
- if (!this.isListening) this.setup();
59
- if (this.landmarks.find((landmark)=>landmark.ref === newLandmark.ref
60
- )) return;
61
- if (this.landmarks.filter((landmark)=>landmark.role === 'main'
62
- ).length > 1) console.error('Page can contain no more than one landmark with the role "main".');
104
+ this.setupIfNeeded();
105
+ if (this.landmarks.find((landmark)=>landmark.ref === newLandmark.ref)) return;
106
+ if (this.landmarks.filter((landmark)=>landmark.role === "main").length > 1) console.error('Page can contain no more than one landmark with the role "main".');
63
107
  if (this.landmarks.length === 0) {
64
108
  this.landmarks = [
65
109
  newLandmark
66
110
  ];
111
+ this.checkLabels(newLandmark.role);
67
112
  return;
68
113
  }
69
114
  // Binary search to insert new landmark based on position in document relative to existing landmarks.
@@ -78,10 +123,10 @@ class $202c109aedff6705$var$LandmarkManager {
78
123
  else end = mid - 1;
79
124
  }
80
125
  this.landmarks.splice(start, 0, newLandmark);
126
+ this.checkLabels(newLandmark.role);
81
127
  }
82
128
  updateLandmark(landmark) {
83
- let index = this.landmarks.findIndex((l)=>l.ref === landmark.ref
84
- );
129
+ let index = this.landmarks.findIndex((l)=>l.ref === landmark.ref);
85
130
  if (index >= 0) {
86
131
  this.landmarks[index] = {
87
132
  ...this.landmarks[index],
@@ -91,9 +136,8 @@ class $202c109aedff6705$var$LandmarkManager {
91
136
  }
92
137
  }
93
138
  removeLandmark(ref) {
94
- this.landmarks = this.landmarks.filter((landmark)=>landmark.ref !== ref
95
- );
96
- if (this.landmarks.length === 0) this.teardown();
139
+ this.landmarks = this.landmarks.filter((landmark)=>landmark.ref !== ref);
140
+ this.teardownIfNeeded();
97
141
  }
98
142
  /**
99
143
  * Warn if there are 2+ landmarks with the same role but no label.
@@ -105,23 +149,17 @@ class $202c109aedff6705$var$LandmarkManager {
105
149
  if (landmarksWithRole.size > 1) {
106
150
  let duplicatesWithoutLabel = [
107
151
  ...landmarksWithRole
108
- ].filter((landmark)=>!landmark.label
109
- );
110
- if (duplicatesWithoutLabel.length > 0) console.warn(`Page contains more than one landmark with the '${role}' role. If two or more landmarks on a page share the same role, all must be labeled with an aria-label or aria-labelledby attribute: `, duplicatesWithoutLabel.map((landmark)=>landmark.ref.current
111
- ));
152
+ ].filter((landmark)=>!landmark.label);
153
+ if (duplicatesWithoutLabel.length > 0) console.warn(`Page contains more than one landmark with the '${role}' role. If two or more landmarks on a page share the same role, all must be labeled with an aria-label or aria-labelledby attribute: `, duplicatesWithoutLabel.map((landmark)=>landmark.ref.current));
112
154
  else {
113
155
  let labels = [
114
156
  ...landmarksWithRole
115
- ].map((landmark)=>landmark.label
116
- );
117
- let duplicateLabels = labels.filter((item, index)=>labels.indexOf(item) !== index
118
- );
157
+ ].map((landmark)=>landmark.label);
158
+ let duplicateLabels = labels.filter((item, index)=>labels.indexOf(item) !== index);
119
159
  duplicateLabels.forEach((label)=>{
120
160
  console.warn(`Page contains more than one landmark with the '${role}' role and '${label}' label. If two or more landmarks on a page share the same role, they must have unique labels: `, [
121
161
  ...landmarksWithRole
122
- ].filter((landmark)=>landmark.label === label
123
- ).map((landmark)=>landmark.ref.current
124
- ));
162
+ ].filter((landmark)=>landmark.label === label).map((landmark)=>landmark.ref.current));
125
163
  });
126
164
  }
127
165
  }
@@ -133,10 +171,9 @@ class $202c109aedff6705$var$LandmarkManager {
133
171
  let landmarkMap = new Map(this.landmarks.map((l)=>[
134
172
  l.ref.current,
135
173
  l
136
- ]
137
- ));
174
+ ]));
138
175
  let currentElement = element;
139
- while(!landmarkMap.has(currentElement) && currentElement !== document.body)currentElement = currentElement.parentElement;
176
+ while(currentElement && !landmarkMap.has(currentElement) && currentElement !== document.body)currentElement = currentElement.parentElement;
140
177
  return landmarkMap.get(currentElement);
141
178
  }
142
179
  /**
@@ -145,14 +182,42 @@ class $202c109aedff6705$var$LandmarkManager {
145
182
  * If not inside a landmark, will return first landmark.
146
183
  * Returns undefined if there are no landmarks.
147
184
  */ getNextLandmark(element, { backward: backward }) {
148
- if (this.landmarks.length === 0) return undefined;
149
185
  let currentLandmark = this.closestLandmark(element);
150
- let nextLandmarkIndex = backward ? -1 : 0;
151
- if (currentLandmark) nextLandmarkIndex = this.landmarks.findIndex((landmark)=>landmark === currentLandmark
152
- ) + (backward ? -1 : 1);
153
- // Wrap if necessary
154
- if (nextLandmarkIndex < 0) nextLandmarkIndex = this.landmarks.length - 1;
155
- else if (nextLandmarkIndex >= this.landmarks.length) nextLandmarkIndex = 0;
186
+ let nextLandmarkIndex = backward ? this.landmarks.length - 1 : 0;
187
+ if (currentLandmark) nextLandmarkIndex = this.landmarks.indexOf(currentLandmark) + (backward ? -1 : 1);
188
+ let wrapIfNeeded = ()=>{
189
+ // When we reach the end of the landmark sequence, fire a custom event that can be listened for by applications.
190
+ // If this event is canceled, we return immediately. This can be used to implement landmark navigation across iframes.
191
+ if (nextLandmarkIndex < 0) {
192
+ if (!element.dispatchEvent(new CustomEvent("react-aria-landmark-navigation", {
193
+ detail: {
194
+ direction: "backward"
195
+ },
196
+ bubbles: true,
197
+ cancelable: true
198
+ }))) return true;
199
+ nextLandmarkIndex = this.landmarks.length - 1;
200
+ } else if (nextLandmarkIndex >= this.landmarks.length) {
201
+ if (!element.dispatchEvent(new CustomEvent("react-aria-landmark-navigation", {
202
+ detail: {
203
+ direction: "forward"
204
+ },
205
+ bubbles: true,
206
+ cancelable: true
207
+ }))) return true;
208
+ nextLandmarkIndex = 0;
209
+ }
210
+ if (nextLandmarkIndex < 0 || nextLandmarkIndex >= this.landmarks.length) return true;
211
+ return false;
212
+ };
213
+ if (wrapIfNeeded()) return undefined;
214
+ // Skip over hidden landmarks.
215
+ let i = nextLandmarkIndex;
216
+ while(this.landmarks[nextLandmarkIndex].ref.current.closest("[aria-hidden=true]")){
217
+ nextLandmarkIndex += backward ? -1 : 1;
218
+ if (wrapIfNeeded()) return undefined;
219
+ if (nextLandmarkIndex === i) break;
220
+ }
156
221
  return this.landmarks[nextLandmarkIndex];
157
222
  }
158
223
  /**
@@ -160,32 +225,42 @@ class $202c109aedff6705$var$LandmarkManager {
160
225
  * If not, focus the landmark itself.
161
226
  * If no landmarks at all, or none with focusable elements, don't move focus.
162
227
  */ f6Handler(e) {
163
- if (e.key === 'F6') {
164
- e.preventDefault();
165
- e.stopPropagation();
166
- let backward = e.shiftKey;
167
- let nextLandmark = this.getNextLandmark(e.target, {
168
- backward: backward
169
- });
170
- // If no landmarks, return
171
- if (!nextLandmark) return;
172
- // If alt key pressed, focus main landmark
173
- if (e.altKey) {
174
- let main = this.getLandmarkByRole('main');
175
- if (main && document.contains(main.ref.current)) this.focusLandmark(main.ref.current);
176
- return;
228
+ if (e.key === "F6") {
229
+ // If alt key pressed, focus main landmark, otherwise navigate forward or backward based on shift key.
230
+ let handled = e.altKey ? this.focusMain() : this.navigate(e.target, e.shiftKey);
231
+ if (handled) {
232
+ e.preventDefault();
233
+ e.stopPropagation();
177
234
  }
178
- // If something was previously focused in the next landmark, then return focus to it
179
- if (nextLandmark.lastFocused) {
180
- let lastFocused = nextLandmark.lastFocused;
181
- if (document.body.contains(lastFocused)) {
182
- lastFocused.focus();
183
- return;
184
- }
235
+ }
236
+ }
237
+ focusMain() {
238
+ let main = this.getLandmarkByRole("main");
239
+ if (main && document.contains(main.ref.current)) {
240
+ this.focusLandmark(main.ref.current, "forward");
241
+ return true;
242
+ }
243
+ return false;
244
+ }
245
+ navigate(from, backward) {
246
+ let nextLandmark = this.getNextLandmark(from, {
247
+ backward: backward
248
+ });
249
+ if (!nextLandmark) return false;
250
+ // If something was previously focused in the next landmark, then return focus to it
251
+ if (nextLandmark.lastFocused) {
252
+ let lastFocused = nextLandmark.lastFocused;
253
+ if (document.body.contains(lastFocused)) {
254
+ lastFocused.focus();
255
+ return true;
185
256
  }
186
- // Otherwise, focus the landmark itself
187
- if (document.contains(nextLandmark.ref.current)) this.focusLandmark(nextLandmark.ref.current);
188
257
  }
258
+ // Otherwise, focus the landmark itself
259
+ if (document.contains(nextLandmark.ref.current)) {
260
+ this.focusLandmark(nextLandmark.ref.current, backward ? "backward" : "forward");
261
+ return true;
262
+ }
263
+ return false;
189
264
  }
190
265
  /**
191
266
  * Sets lastFocused for a landmark, if focus is moved within that landmark.
@@ -214,57 +289,111 @@ class $202c109aedff6705$var$LandmarkManager {
214
289
  if (closestPreviousLandmark && closestPreviousLandmark.ref.current === previousFocusedElement) closestPreviousLandmark.blur();
215
290
  }
216
291
  }
292
+ createLandmarkController() {
293
+ let instance = this;
294
+ instance.refCount++;
295
+ instance.setupIfNeeded();
296
+ return {
297
+ navigate (direction, opts) {
298
+ return instance.navigate((opts === null || opts === void 0 ? void 0 : opts.from) || document.activeElement, direction === "backward");
299
+ },
300
+ focusNext (opts) {
301
+ return instance.navigate((opts === null || opts === void 0 ? void 0 : opts.from) || document.activeElement, false);
302
+ },
303
+ focusPrevious (opts) {
304
+ return instance.navigate((opts === null || opts === void 0 ? void 0 : opts.from) || document.activeElement, true);
305
+ },
306
+ focusMain () {
307
+ return instance.focusMain();
308
+ },
309
+ dispose () {
310
+ instance.refCount--;
311
+ instance.teardownIfNeeded();
312
+ instance = null;
313
+ }
314
+ };
315
+ }
316
+ registerLandmark(landmark) {
317
+ if (this.landmarks.find((l)=>l.ref === landmark.ref)) this.updateLandmark(landmark);
318
+ else this.addLandmark(landmark);
319
+ return ()=>this.removeLandmark(landmark.ref);
320
+ }
217
321
  constructor(){
218
- this.landmarks = [];
219
- this.isListening = false;
322
+ (0, ($parcel$interopDefault($8Ore6$swchelperslib_define_propertyjs)))(this, "landmarks", []);
323
+ (0, ($parcel$interopDefault($8Ore6$swchelperslib_define_propertyjs)))(this, "isListening", false);
324
+ (0, ($parcel$interopDefault($8Ore6$swchelperslib_define_propertyjs)))(this, "refCount", 0);
325
+ (0, ($parcel$interopDefault($8Ore6$swchelperslib_define_propertyjs)))(this, "version", $202c109aedff6705$var$LANDMARK_API_VERSION);
220
326
  this.f6Handler = this.f6Handler.bind(this);
221
327
  this.focusinHandler = this.focusinHandler.bind(this);
222
328
  this.focusoutHandler = this.focusoutHandler.bind(this);
223
329
  }
224
330
  }
331
+ function $202c109aedff6705$export$f50151dbd51cd1d9() {
332
+ // Get the current landmark manager and create a controller using it.
333
+ let instance = $202c109aedff6705$var$getLandmarkManager();
334
+ let controller = instance.createLandmarkController();
335
+ let unsubscribe = $202c109aedff6705$var$subscribe(()=>{
336
+ // If the landmark manager changes, dispose the old
337
+ // controller and create a new one.
338
+ controller.dispose();
339
+ instance = $202c109aedff6705$var$getLandmarkManager();
340
+ controller = instance.createLandmarkController();
341
+ });
342
+ // Return a wrapper that proxies requests to the current controller instance.
343
+ return {
344
+ navigate (direction, opts) {
345
+ return controller.navigate(direction, opts);
346
+ },
347
+ focusNext (opts) {
348
+ return controller.focusNext(opts);
349
+ },
350
+ focusPrevious (opts) {
351
+ return controller.focusPrevious(opts);
352
+ },
353
+ focusMain () {
354
+ return controller.focusMain();
355
+ },
356
+ dispose () {
357
+ controller.dispose();
358
+ unsubscribe();
359
+ controller = null;
360
+ instance = null;
361
+ }
362
+ };
363
+ }
225
364
  function $202c109aedff6705$export$4cc632584fd87fae(props, ref) {
226
- const { role: role , 'aria-label': ariaLabel , 'aria-labelledby': ariaLabelledby } = props;
227
- let manager = $202c109aedff6705$var$LandmarkManager.getInstance();
365
+ const { role: role , "aria-label": ariaLabel , "aria-labelledby": ariaLabelledby , focus: focus } = props;
366
+ let manager = $202c109aedff6705$var$useLandmarkManager();
228
367
  let label = ariaLabel || ariaLabelledby;
229
- let [isLandmarkFocused, setIsLandmarkFocused] = $8Ore6$react.useState(false);
230
- let focus = $8Ore6$react.useCallback(()=>{
368
+ let [isLandmarkFocused, setIsLandmarkFocused] = (0, $8Ore6$react.useState)(false);
369
+ let defaultFocus = (0, $8Ore6$react.useCallback)(()=>{
231
370
  setIsLandmarkFocused(true);
232
371
  }, [
233
372
  setIsLandmarkFocused
234
373
  ]);
235
- let blur = $8Ore6$react.useCallback(()=>{
374
+ let blur = (0, $8Ore6$react.useCallback)(()=>{
236
375
  setIsLandmarkFocused(false);
237
376
  }, [
238
377
  setIsLandmarkFocused
239
378
  ]);
240
- $8Ore6$reactariautils.useLayoutEffect(()=>{
241
- manager.addLandmark({
242
- ref: ref,
243
- role: role,
244
- label: label,
245
- focus: focus,
246
- blur: blur
247
- });
248
- return ()=>{
249
- manager.removeLandmark(ref);
250
- };
251
- // eslint-disable-next-line react-hooks/exhaustive-deps
252
- }, []);
253
- $8Ore6$reactariautils.useLayoutEffect(()=>{
254
- manager.updateLandmark({
379
+ (0, $8Ore6$reactariautils.useLayoutEffect)(()=>{
380
+ return manager.registerLandmark({
255
381
  ref: ref,
256
382
  label: label,
257
383
  role: role,
258
- focus: focus,
384
+ focus: focus || defaultFocus,
259
385
  blur: blur
260
386
  });
261
- // eslint-disable-next-line react-hooks/exhaustive-deps
262
387
  }, [
388
+ manager,
263
389
  label,
264
390
  ref,
265
- role
391
+ role,
392
+ focus,
393
+ defaultFocus,
394
+ blur
266
395
  ]);
267
- $8Ore6$react.useEffect(()=>{
396
+ (0, $8Ore6$react.useEffect)(()=>{
268
397
  if (isLandmarkFocused) ref.current.focus();
269
398
  }, [
270
399
  isLandmarkFocused,
@@ -273,7 +402,9 @@ function $202c109aedff6705$export$4cc632584fd87fae(props, ref) {
273
402
  return {
274
403
  landmarkProps: {
275
404
  role: role,
276
- tabIndex: isLandmarkFocused ? -1 : undefined
405
+ tabIndex: isLandmarkFocused ? -1 : undefined,
406
+ "aria-label": ariaLabel,
407
+ "aria-labelledby": ariaLabelledby
277
408
  }
278
409
  };
279
410
  }
package/dist/main.js.map CHANGED
@@ -1 +1 @@
1
- {"mappings":"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","sources":["packages/@react-aria/landmark/src/index.ts","packages/@react-aria/landmark/src/useLandmark.ts"],"sourcesContent":["/*\n * Copyright 2022 Adobe. All rights reserved.\n * This file is licensed to you under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License. You may obtain a copy\n * of the License at http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under\n * the License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\n * OF ANY KIND, either express or implied. See the License for the specific language\n * governing permissions and limitations under the License.\n */\n\nexport type {AriaLandmarkRole, AriaLandmarkProps, LandmarkAria} from './useLandmark';\nexport {useLandmark} from './useLandmark';\n","/*\n * Copyright 2022 Adobe. All rights reserved.\n * This file is licensed to you under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License. You may obtain a copy\n * of the License at http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under\n * the License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\n * OF ANY KIND, either express or implied. See the License for the specific language\n * governing permissions and limitations under the License.\n */\n\nimport {AriaLabelingProps, DOMAttributes, FocusableElement} from '@react-types/shared';\nimport {MutableRefObject, useCallback, useEffect, useState} from 'react';\nimport {useLayoutEffect} from '@react-aria/utils';\n\nexport type AriaLandmarkRole = 'main' | 'region' | 'search' | 'navigation' | 'form' | 'banner' | 'contentinfo' | 'complementary';\n\nexport interface AriaLandmarkProps extends AriaLabelingProps {\n role: AriaLandmarkRole\n}\n\nexport interface LandmarkAria {\n landmarkProps: DOMAttributes\n}\n\ntype Landmark = {\n ref: MutableRefObject<Element>,\n role: AriaLandmarkRole,\n label?: string,\n lastFocused?: FocusableElement,\n focus: () => void,\n blur: () => void\n};\n\nclass LandmarkManager {\n private landmarks: Array<Landmark> = [];\n private static instance: LandmarkManager;\n private isListening = false;\n\n private constructor() {\n this.f6Handler = this.f6Handler.bind(this);\n this.focusinHandler = this.focusinHandler.bind(this);\n this.focusoutHandler = this.focusoutHandler.bind(this);\n }\n\n public static getInstance(): LandmarkManager {\n if (!LandmarkManager.instance) {\n LandmarkManager.instance = new LandmarkManager();\n }\n\n return LandmarkManager.instance;\n }\n\n private setup() {\n document.addEventListener('keydown', this.f6Handler, {capture: true});\n document.addEventListener('focusin', this.focusinHandler, {capture: true});\n document.addEventListener('focusout', this.focusoutHandler, {capture: true});\n this.isListening = true;\n }\n\n private teardown() {\n document.removeEventListener('keydown', this.f6Handler, {capture: true});\n document.removeEventListener('focusin', this.focusinHandler, {capture: true});\n document.removeEventListener('focusout', this.focusoutHandler, {capture: true});\n this.isListening = false;\n }\n\n private focusLandmark(landmark: Element) {\n this.landmarks.find(l => l.ref.current === landmark)?.focus();\n }\n\n /**\n * Return set of landmarks with a specific role.\n */\n public getLandmarksByRole(role: AriaLandmarkRole) {\n return new Set(this.landmarks.filter(l => l.role === role));\n }\n\n /**\n * Return first landmark with a specific role.\n */\n public getLandmarkByRole(role: AriaLandmarkRole) {\n return this.landmarks.find(l => l.role === role);\n }\n\n public addLandmark(newLandmark: Landmark) {\n if (!this.isListening) {\n this.setup();\n }\n if (this.landmarks.find(landmark => landmark.ref === newLandmark.ref)) {\n return;\n }\n\n if (this.landmarks.filter(landmark => landmark.role === 'main').length > 1) {\n console.error('Page can contain no more than one landmark with the role \"main\".');\n }\n\n if (this.landmarks.length === 0) {\n this.landmarks = [newLandmark];\n return;\n }\n\n\n // Binary search to insert new landmark based on position in document relative to existing landmarks.\n // https://developer.mozilla.org/en-US/docs/Web/API/Node/compareDocumentPosition\n let start = 0;\n let end = this.landmarks.length - 1;\n while (start <= end) {\n let mid = Math.floor((start + end) / 2);\n let comparedPosition = newLandmark.ref.current.compareDocumentPosition(this.landmarks[mid].ref.current as Node);\n let isNewAfterExisting = Boolean((comparedPosition & Node.DOCUMENT_POSITION_PRECEDING) || (comparedPosition & Node.DOCUMENT_POSITION_CONTAINS));\n\n if (isNewAfterExisting) {\n start = mid + 1;\n } else {\n end = mid - 1;\n }\n }\n\n this.landmarks.splice(start, 0, newLandmark);\n }\n\n public updateLandmark(landmark: Pick<Landmark, 'ref'> & Partial<Landmark>) {\n let index = this.landmarks.findIndex(l => l.ref === landmark.ref);\n if (index >= 0) {\n this.landmarks[index] = {...this.landmarks[index], ...landmark};\n this.checkLabels(this.landmarks[index].role);\n }\n }\n\n public removeLandmark(ref: MutableRefObject<Element>) {\n this.landmarks = this.landmarks.filter(landmark => landmark.ref !== ref);\n if (this.landmarks.length === 0) {\n this.teardown();\n }\n }\n\n /**\n * Warn if there are 2+ landmarks with the same role but no label.\n * Labels for landmarks with the same role must also be unique.\n *\n * See https://www.w3.org/WAI/ARIA/apg/practices/landmark-regions/.\n */\n private checkLabels(role: AriaLandmarkRole) {\n let landmarksWithRole = this.getLandmarksByRole(role);\n if (landmarksWithRole.size > 1) {\n let duplicatesWithoutLabel = [...landmarksWithRole].filter(landmark => !landmark.label);\n if (duplicatesWithoutLabel.length > 0) {\n console.warn(\n `Page contains more than one landmark with the '${role}' role. If two or more landmarks on a page share the same role, all must be labeled with an aria-label or aria-labelledby attribute: `,\n duplicatesWithoutLabel.map(landmark => landmark.ref.current)\n );\n } else {\n let labels = [...landmarksWithRole].map(landmark => landmark.label);\n let duplicateLabels = labels.filter((item, index) => labels.indexOf(item) !== index);\n\n duplicateLabels.forEach((label) => {\n console.warn(\n `Page contains more than one landmark with the '${role}' role and '${label}' label. If two or more landmarks on a page share the same role, they must have unique labels: `,\n [...landmarksWithRole].filter(landmark => landmark.label === label).map(landmark => landmark.ref.current)\n );\n });\n }\n }\n }\n\n /**\n * Get the landmark that is the closest parent in the DOM.\n * Returns undefined if no parent is a landmark.\n */\n private closestLandmark(element: Element) {\n let landmarkMap = new Map(this.landmarks.map(l => [l.ref.current, l]));\n let currentElement = element;\n while (!landmarkMap.has(currentElement) && currentElement !== document.body) {\n currentElement = currentElement.parentElement;\n }\n return landmarkMap.get(currentElement);\n }\n\n /**\n * Gets the next landmark, in DOM focus order, or previous if backwards is specified.\n * If last landmark, next should be the first landmark.\n * If not inside a landmark, will return first landmark.\n * Returns undefined if there are no landmarks.\n */\n public getNextLandmark(element: Element, {backward}: {backward?: boolean }) {\n if (this.landmarks.length === 0) {\n return undefined;\n }\n\n let currentLandmark = this.closestLandmark(element);\n let nextLandmarkIndex = backward ? -1 : 0;\n if (currentLandmark) {\n nextLandmarkIndex = this.landmarks.findIndex(landmark => landmark === currentLandmark) + (backward ? -1 : 1);\n }\n\n // Wrap if necessary\n if (nextLandmarkIndex < 0) {\n nextLandmarkIndex = this.landmarks.length - 1;\n } else if (nextLandmarkIndex >= this.landmarks.length) {\n nextLandmarkIndex = 0;\n }\n\n return this.landmarks[nextLandmarkIndex];\n }\n\n /**\n * Look at next landmark. If an element was previously focused inside, restore focus there.\n * If not, focus the landmark itself.\n * If no landmarks at all, or none with focusable elements, don't move focus.\n */\n public f6Handler(e: KeyboardEvent) {\n if (e.key === 'F6') {\n e.preventDefault();\n e.stopPropagation();\n\n let backward = e.shiftKey;\n let nextLandmark = this.getNextLandmark(e.target as Element, {backward});\n\n // If no landmarks, return\n if (!nextLandmark) {\n return;\n }\n\n // If alt key pressed, focus main landmark\n if (e.altKey) {\n let main = this.getLandmarkByRole('main');\n if (main && document.contains(main.ref.current)) {\n this.focusLandmark(main.ref.current);\n }\n return;\n }\n\n // If something was previously focused in the next landmark, then return focus to it\n if (nextLandmark.lastFocused) {\n let lastFocused = nextLandmark.lastFocused;\n if (document.body.contains(lastFocused)) {\n lastFocused.focus();\n return;\n }\n }\n\n // Otherwise, focus the landmark itself\n if (document.contains(nextLandmark.ref.current)) {\n this.focusLandmark(nextLandmark.ref.current);\n }\n }\n }\n\n /**\n * Sets lastFocused for a landmark, if focus is moved within that landmark.\n * Lets the last focused landmark know it was blurred if something else is focused.\n */\n public focusinHandler(e: FocusEvent) {\n let currentLandmark = this.closestLandmark(e.target as Element);\n if (currentLandmark && currentLandmark.ref.current !== e.target) {\n this.updateLandmark({ref: currentLandmark.ref, lastFocused: e.target as FocusableElement});\n }\n let previousFocusedElement = e.relatedTarget as Element;\n if (previousFocusedElement) {\n let closestPreviousLandmark = this.closestLandmark(previousFocusedElement);\n if (closestPreviousLandmark && closestPreviousLandmark.ref.current === previousFocusedElement) {\n closestPreviousLandmark.blur();\n }\n }\n }\n\n /**\n * Track if the focus is lost to the body. If it is, do cleanup on the landmark that last had focus.\n */\n public focusoutHandler(e: FocusEvent) {\n let previousFocusedElement = e.target as Element;\n let nextFocusedElement = e.relatedTarget;\n // the === document seems to be a jest thing for focus to go there on generic blur event such as landmark.blur();\n // browsers appear to send focus instead to document.body and the relatedTarget is null when that happens\n if (!nextFocusedElement || nextFocusedElement === document) {\n let closestPreviousLandmark = this.closestLandmark(previousFocusedElement);\n if (closestPreviousLandmark && closestPreviousLandmark.ref.current === previousFocusedElement) {\n closestPreviousLandmark.blur();\n }\n }\n }\n}\n\n/**\n * Provides landmark navigation in an application. Call this with a role and label to register a landmark navigable with F6.\n * @param props - Props for the landmark.\n * @param ref - Ref to the landmark.\n */\nexport function useLandmark(props: AriaLandmarkProps, ref: MutableRefObject<FocusableElement>): LandmarkAria {\n const {\n role,\n 'aria-label': ariaLabel,\n 'aria-labelledby': ariaLabelledby\n } = props;\n let manager = LandmarkManager.getInstance();\n let label = ariaLabel || ariaLabelledby;\n let [isLandmarkFocused, setIsLandmarkFocused] = useState(false);\n\n let focus = useCallback(() => {\n setIsLandmarkFocused(true);\n }, [setIsLandmarkFocused]);\n\n let blur = useCallback(() => {\n setIsLandmarkFocused(false);\n }, [setIsLandmarkFocused]);\n\n useLayoutEffect(() => {\n manager.addLandmark({ref, role, label, focus, blur});\n\n return () => {\n manager.removeLandmark(ref);\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useLayoutEffect(() => {\n manager.updateLandmark({ref, label, role, focus, blur});\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [label, ref, role]);\n\n useEffect(() => {\n if (isLandmarkFocused) {\n ref.current.focus();\n }\n }, [isLandmarkFocused, ref]);\n\n return {\n landmarkProps: {\n role,\n tabIndex: isLandmarkFocused ? -1 : undefined\n }\n };\n}\n"],"names":[],"version":3,"file":"main.js.map"}
1
+ {"mappings":"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","sources":["packages/@react-aria/landmark/src/index.ts","packages/@react-aria/landmark/src/useLandmark.ts"],"sourcesContent":["/*\n * Copyright 2022 Adobe. All rights reserved.\n * This file is licensed to you under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License. You may obtain a copy\n * of the License at http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under\n * the License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\n * OF ANY KIND, either express or implied. See the License for the specific language\n * governing permissions and limitations under the License.\n */\n\nexport type {AriaLandmarkRole, AriaLandmarkProps, LandmarkAria, LandmarkController} from './useLandmark';\nexport {useLandmark, createLandmarkController} from './useLandmark';\n","/*\n * Copyright 2022 Adobe. All rights reserved.\n * This file is licensed to you under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License. You may obtain a copy\n * of the License at http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software distributed under\n * the License is distributed on an \"AS IS\" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS\n * OF ANY KIND, either express or implied. See the License for the specific language\n * governing permissions and limitations under the License.\n */\n\nimport {AriaLabelingProps, DOMAttributes, FocusableElement} from '@react-types/shared';\nimport {MutableRefObject, useCallback, useEffect, useState} from 'react';\nimport {useLayoutEffect} from '@react-aria/utils';\nimport {useSyncExternalStore} from 'use-sync-external-store/shim';\n\nexport type AriaLandmarkRole = 'main' | 'region' | 'search' | 'navigation' | 'form' | 'banner' | 'contentinfo' | 'complementary';\n\nexport interface AriaLandmarkProps extends AriaLabelingProps {\n role: AriaLandmarkRole,\n focus?: (direction: 'forward' | 'backward') => void\n}\n\nexport interface LandmarkAria {\n landmarkProps: DOMAttributes\n}\n\n// Increment this version number whenever the\n// LandmarkManagerApi or Landmark interfaces change.\nconst LANDMARK_API_VERSION = 1;\n\n// Minimal API for LandmarkManager that must continue to work between versions.\n// Changes to this interface are considered breaking. New methods/properties are\n// safe to add, but changes or removals are not allowed (same as public APIs).\ninterface LandmarkManagerApi {\n version: number,\n createLandmarkController(): LandmarkController,\n registerLandmark(landmark: Landmark): () => void\n}\n\n// Changes to this interface are considered breaking.\n// New properties MUST be optional so that registering a landmark\n// from an older version of useLandmark against a newer version of\n// LandmarkManager does not crash.\ninterface Landmark {\n ref: MutableRefObject<Element>,\n role: AriaLandmarkRole,\n label?: string,\n lastFocused?: FocusableElement,\n focus: (direction: 'forward' | 'backward') => void,\n blur: () => void\n}\n\nexport interface LandmarkControllerOptions {\n /**\n * The element from which to start navigating.\n * @default document.activeElement\n */\n from?: Element\n}\n\n/** A LandmarkController allows programmatic navigation of landmarks. */\nexport interface LandmarkController {\n /** Moves focus to the next landmark. */\n focusNext(opts?: LandmarkControllerOptions): boolean,\n /** Moves focus to the previous landmark. */\n focusPrevious(opts?: LandmarkControllerOptions): boolean,\n /** Moves focus to the main landmark. */\n focusMain(): boolean,\n /** Moves focus either forward or backward in the landmark sequence. */\n navigate(direction: 'forward' | 'backward', opts?: LandmarkControllerOptions): boolean,\n /**\n * Disposes the landmark controller. When no landmarks are registered, and no\n * controllers are active, the landmark keyboard listeners are removed from the page.\n */\n dispose(): void\n}\n\n// Symbol under which the singleton landmark manager instance is attached to the document.\nconst landmarkSymbol = Symbol.for('react-aria-landmark-manager');\n\nfunction subscribe(fn: () => void) {\n document.addEventListener('react-aria-landmark-manager-change', fn);\n return () => document.removeEventListener('react-aria-landmark-manager-change', fn);\n}\n\nfunction getLandmarkManager(): LandmarkManagerApi {\n // Reuse an existing instance if it has the same or greater version.\n let instance = document[landmarkSymbol];\n if (instance && instance.version >= LANDMARK_API_VERSION) {\n return instance;\n }\n\n // Otherwise, create a new instance and dispatch an event so anything using the existing\n // instance updates and re-registers their landmarks with the new one.\n document[landmarkSymbol] = new LandmarkManager();\n document.dispatchEvent(new CustomEvent('react-aria-landmark-manager-change'));\n return document[landmarkSymbol];\n}\n\n// Subscribes a React component to the current landmark manager instance.\nfunction useLandmarkManager(): LandmarkManagerApi {\n return useSyncExternalStore(subscribe, getLandmarkManager);\n}\n\nclass LandmarkManager implements LandmarkManagerApi {\n private landmarks: Array<Landmark> = [];\n private isListening = false;\n private refCount = 0;\n public version = LANDMARK_API_VERSION;\n\n constructor() {\n this.f6Handler = this.f6Handler.bind(this);\n this.focusinHandler = this.focusinHandler.bind(this);\n this.focusoutHandler = this.focusoutHandler.bind(this);\n }\n\n private setupIfNeeded() {\n if (this.isListening) {\n return;\n }\n document.addEventListener('keydown', this.f6Handler, {capture: true});\n document.addEventListener('focusin', this.focusinHandler, {capture: true});\n document.addEventListener('focusout', this.focusoutHandler, {capture: true});\n this.isListening = true;\n }\n\n private teardownIfNeeded() {\n if (!this.isListening || this.landmarks.length > 0 || this.refCount > 0) {\n return;\n }\n document.removeEventListener('keydown', this.f6Handler, {capture: true});\n document.removeEventListener('focusin', this.focusinHandler, {capture: true});\n document.removeEventListener('focusout', this.focusoutHandler, {capture: true});\n this.isListening = false;\n }\n\n private focusLandmark(landmark: Element, direction: 'forward' | 'backward') {\n this.landmarks.find(l => l.ref.current === landmark)?.focus(direction);\n }\n\n /**\n * Return set of landmarks with a specific role.\n */\n private getLandmarksByRole(role: AriaLandmarkRole) {\n return new Set(this.landmarks.filter(l => l.role === role));\n }\n\n /**\n * Return first landmark with a specific role.\n */\n private getLandmarkByRole(role: AriaLandmarkRole) {\n return this.landmarks.find(l => l.role === role);\n }\n\n private addLandmark(newLandmark: Landmark) {\n this.setupIfNeeded();\n if (this.landmarks.find(landmark => landmark.ref === newLandmark.ref)) {\n return;\n }\n\n if (this.landmarks.filter(landmark => landmark.role === 'main').length > 1) {\n console.error('Page can contain no more than one landmark with the role \"main\".');\n }\n\n if (this.landmarks.length === 0) {\n this.landmarks = [newLandmark];\n this.checkLabels(newLandmark.role);\n return;\n }\n\n\n // Binary search to insert new landmark based on position in document relative to existing landmarks.\n // https://developer.mozilla.org/en-US/docs/Web/API/Node/compareDocumentPosition\n let start = 0;\n let end = this.landmarks.length - 1;\n while (start <= end) {\n let mid = Math.floor((start + end) / 2);\n let comparedPosition = newLandmark.ref.current.compareDocumentPosition(this.landmarks[mid].ref.current as Node);\n let isNewAfterExisting = Boolean((comparedPosition & Node.DOCUMENT_POSITION_PRECEDING) || (comparedPosition & Node.DOCUMENT_POSITION_CONTAINS));\n\n if (isNewAfterExisting) {\n start = mid + 1;\n } else {\n end = mid - 1;\n }\n }\n\n this.landmarks.splice(start, 0, newLandmark);\n this.checkLabels(newLandmark.role);\n }\n\n private updateLandmark(landmark: Pick<Landmark, 'ref'> & Partial<Landmark>) {\n let index = this.landmarks.findIndex(l => l.ref === landmark.ref);\n if (index >= 0) {\n this.landmarks[index] = {...this.landmarks[index], ...landmark};\n this.checkLabels(this.landmarks[index].role);\n }\n }\n\n private removeLandmark(ref: MutableRefObject<Element>) {\n this.landmarks = this.landmarks.filter(landmark => landmark.ref !== ref);\n this.teardownIfNeeded();\n }\n\n /**\n * Warn if there are 2+ landmarks with the same role but no label.\n * Labels for landmarks with the same role must also be unique.\n *\n * See https://www.w3.org/WAI/ARIA/apg/practices/landmark-regions/.\n */\n private checkLabels(role: AriaLandmarkRole) {\n let landmarksWithRole = this.getLandmarksByRole(role);\n if (landmarksWithRole.size > 1) {\n let duplicatesWithoutLabel = [...landmarksWithRole].filter(landmark => !landmark.label);\n if (duplicatesWithoutLabel.length > 0) {\n console.warn(\n `Page contains more than one landmark with the '${role}' role. If two or more landmarks on a page share the same role, all must be labeled with an aria-label or aria-labelledby attribute: `,\n duplicatesWithoutLabel.map(landmark => landmark.ref.current)\n );\n } else {\n let labels = [...landmarksWithRole].map(landmark => landmark.label);\n let duplicateLabels = labels.filter((item, index) => labels.indexOf(item) !== index);\n\n duplicateLabels.forEach((label) => {\n console.warn(\n `Page contains more than one landmark with the '${role}' role and '${label}' label. If two or more landmarks on a page share the same role, they must have unique labels: `,\n [...landmarksWithRole].filter(landmark => landmark.label === label).map(landmark => landmark.ref.current)\n );\n });\n }\n }\n }\n\n /**\n * Get the landmark that is the closest parent in the DOM.\n * Returns undefined if no parent is a landmark.\n */\n private closestLandmark(element: Element) {\n let landmarkMap = new Map(this.landmarks.map(l => [l.ref.current, l]));\n let currentElement = element;\n while (currentElement && !landmarkMap.has(currentElement) && currentElement !== document.body) {\n currentElement = currentElement.parentElement;\n }\n return landmarkMap.get(currentElement);\n }\n\n /**\n * Gets the next landmark, in DOM focus order, or previous if backwards is specified.\n * If last landmark, next should be the first landmark.\n * If not inside a landmark, will return first landmark.\n * Returns undefined if there are no landmarks.\n */\n private getNextLandmark(element: Element, {backward}: {backward?: boolean }) {\n let currentLandmark = this.closestLandmark(element);\n let nextLandmarkIndex = backward ? this.landmarks.length - 1 : 0;\n if (currentLandmark) {\n nextLandmarkIndex = this.landmarks.indexOf(currentLandmark) + (backward ? -1 : 1);\n }\n\n let wrapIfNeeded = () => {\n // When we reach the end of the landmark sequence, fire a custom event that can be listened for by applications.\n // If this event is canceled, we return immediately. This can be used to implement landmark navigation across iframes.\n if (nextLandmarkIndex < 0) {\n if (!element.dispatchEvent(new CustomEvent('react-aria-landmark-navigation', {detail: {direction: 'backward'}, bubbles: true, cancelable: true}))) {\n return true;\n }\n\n nextLandmarkIndex = this.landmarks.length - 1;\n } else if (nextLandmarkIndex >= this.landmarks.length) {\n if (!element.dispatchEvent(new CustomEvent('react-aria-landmark-navigation', {detail: {direction: 'forward'}, bubbles: true, cancelable: true}))) {\n return true;\n }\n\n nextLandmarkIndex = 0;\n }\n\n if (nextLandmarkIndex < 0 || nextLandmarkIndex >= this.landmarks.length) {\n return true;\n }\n\n return false;\n };\n\n if (wrapIfNeeded()) {\n return undefined;\n }\n\n // Skip over hidden landmarks.\n let i = nextLandmarkIndex;\n while (this.landmarks[nextLandmarkIndex].ref.current.closest('[aria-hidden=true]')) {\n nextLandmarkIndex += backward ? -1 : 1;\n if (wrapIfNeeded()) {\n return undefined;\n }\n\n if (nextLandmarkIndex === i) {\n break;\n }\n }\n\n return this.landmarks[nextLandmarkIndex];\n }\n\n /**\n * Look at next landmark. If an element was previously focused inside, restore focus there.\n * If not, focus the landmark itself.\n * If no landmarks at all, or none with focusable elements, don't move focus.\n */\n private f6Handler(e: KeyboardEvent) {\n if (e.key === 'F6') {\n // If alt key pressed, focus main landmark, otherwise navigate forward or backward based on shift key.\n let handled = e.altKey ? this.focusMain() : this.navigate(e.target as Element, e.shiftKey);\n if (handled) {\n e.preventDefault();\n e.stopPropagation();\n }\n }\n }\n\n private focusMain() {\n let main = this.getLandmarkByRole('main');\n if (main && document.contains(main.ref.current)) {\n this.focusLandmark(main.ref.current, 'forward');\n return true;\n }\n\n return false;\n }\n\n private navigate(from: Element, backward: boolean) {\n let nextLandmark = this.getNextLandmark(from, {\n backward\n });\n\n if (!nextLandmark) {\n return false;\n }\n\n // If something was previously focused in the next landmark, then return focus to it\n if (nextLandmark.lastFocused) {\n let lastFocused = nextLandmark.lastFocused;\n if (document.body.contains(lastFocused)) {\n lastFocused.focus();\n return true;\n }\n }\n\n // Otherwise, focus the landmark itself\n if (document.contains(nextLandmark.ref.current)) {\n this.focusLandmark(nextLandmark.ref.current, backward ? 'backward' : 'forward');\n return true;\n }\n\n return false;\n }\n\n /**\n * Sets lastFocused for a landmark, if focus is moved within that landmark.\n * Lets the last focused landmark know it was blurred if something else is focused.\n */\n private focusinHandler(e: FocusEvent) {\n let currentLandmark = this.closestLandmark(e.target as Element);\n if (currentLandmark && currentLandmark.ref.current !== e.target) {\n this.updateLandmark({ref: currentLandmark.ref, lastFocused: e.target as FocusableElement});\n }\n let previousFocusedElement = e.relatedTarget as Element;\n if (previousFocusedElement) {\n let closestPreviousLandmark = this.closestLandmark(previousFocusedElement);\n if (closestPreviousLandmark && closestPreviousLandmark.ref.current === previousFocusedElement) {\n closestPreviousLandmark.blur();\n }\n }\n }\n\n /**\n * Track if the focus is lost to the body. If it is, do cleanup on the landmark that last had focus.\n */\n private focusoutHandler(e: FocusEvent) {\n let previousFocusedElement = e.target as Element;\n let nextFocusedElement = e.relatedTarget;\n // the === document seems to be a jest thing for focus to go there on generic blur event such as landmark.blur();\n // browsers appear to send focus instead to document.body and the relatedTarget is null when that happens\n if (!nextFocusedElement || nextFocusedElement === document) {\n let closestPreviousLandmark = this.closestLandmark(previousFocusedElement);\n if (closestPreviousLandmark && closestPreviousLandmark.ref.current === previousFocusedElement) {\n closestPreviousLandmark.blur();\n }\n }\n }\n\n public createLandmarkController(): LandmarkController {\n let instance = this;\n instance.refCount++;\n instance.setupIfNeeded();\n return {\n navigate(direction, opts) {\n return instance.navigate(opts?.from || document.activeElement, direction === 'backward');\n },\n focusNext(opts) {\n return instance.navigate(opts?.from || document.activeElement, false);\n },\n focusPrevious(opts) {\n return instance.navigate(opts?.from || document.activeElement, true);\n },\n focusMain() {\n return instance.focusMain();\n },\n dispose() {\n instance.refCount--;\n instance.teardownIfNeeded();\n instance = null;\n }\n };\n }\n\n public registerLandmark(landmark: Landmark): () => void {\n if (this.landmarks.find(l => l.ref === landmark.ref)) {\n this.updateLandmark(landmark);\n } else {\n this.addLandmark(landmark);\n }\n\n return () => this.removeLandmark(landmark.ref);\n }\n}\n\n/** Creates a LandmarkController, which allows programmatic navigation of landmarks. */\nexport function createLandmarkController(): LandmarkController {\n // Get the current landmark manager and create a controller using it.\n let instance = getLandmarkManager();\n let controller = instance.createLandmarkController();\n\n let unsubscribe = subscribe(() => {\n // If the landmark manager changes, dispose the old\n // controller and create a new one.\n controller.dispose();\n instance = getLandmarkManager();\n controller = instance.createLandmarkController();\n });\n\n // Return a wrapper that proxies requests to the current controller instance.\n return {\n navigate(direction, opts) {\n return controller.navigate(direction, opts);\n },\n focusNext(opts) {\n return controller.focusNext(opts);\n },\n focusPrevious(opts) {\n return controller.focusPrevious(opts);\n },\n focusMain() {\n return controller.focusMain();\n },\n dispose() {\n controller.dispose();\n unsubscribe();\n controller = null;\n instance = null;\n }\n };\n}\n\n/**\n * Provides landmark navigation in an application. Call this with a role and label to register a landmark navigable with F6.\n * @param props - Props for the landmark.\n * @param ref - Ref to the landmark.\n */\nexport function useLandmark(props: AriaLandmarkProps, ref: MutableRefObject<FocusableElement>): LandmarkAria {\n const {\n role,\n 'aria-label': ariaLabel,\n 'aria-labelledby': ariaLabelledby,\n focus\n } = props;\n let manager = useLandmarkManager();\n let label = ariaLabel || ariaLabelledby;\n let [isLandmarkFocused, setIsLandmarkFocused] = useState(false);\n\n let defaultFocus = useCallback(() => {\n setIsLandmarkFocused(true);\n }, [setIsLandmarkFocused]);\n\n let blur = useCallback(() => {\n setIsLandmarkFocused(false);\n }, [setIsLandmarkFocused]);\n\n useLayoutEffect(() => {\n return manager.registerLandmark({ref, label, role, focus: focus || defaultFocus, blur});\n }, [manager, label, ref, role, focus, defaultFocus, blur]);\n\n useEffect(() => {\n if (isLandmarkFocused) {\n ref.current.focus();\n }\n }, [isLandmarkFocused, ref]);\n\n return {\n landmarkProps: {\n role,\n tabIndex: isLandmarkFocused ? -1 : undefined,\n 'aria-label': ariaLabel,\n 'aria-labelledby': ariaLabelledby\n }\n };\n}\n"],"names":[],"version":3,"file":"main.js.map"}