Skip to content

Commit

Permalink
Further improve tabbable performance (#1750)
Browse files Browse the repository at this point in the history
* improve tabbable performance

* improve tabbable performance

* add PR #

* prettier

* change to getSlottedChildrenOutsideRootElement

* prettier
  • Loading branch information
KonnorRogers authored Dec 1, 2023
1 parent 3e38da2 commit dd27db5
Show file tree
Hide file tree
Showing 2 changed files with 41 additions and 28 deletions.
2 changes: 2 additions & 0 deletions docs/pages/resources/changelog.md
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,8 @@ New versions of Shoelace are released as-needed and generally occur when a criti

## Next

- Fixed focus trapping not scrolling elements into view. [#1750]
- Fixed more performance issues with focus trapping performance. [#1750]
- Added the `hover-bridge` feature to `<sl-popup>` to support better tooltip accessibility [#1734]
- Fixed a bug in `<sl-input>` and `<sl-textarea>` that made it work differently from `<input>` and `<textarea>` when using defaults [#1746]
- Improved the accessibility of `<sl-tooltip>` so they persist when hovering over the tooltip and dismiss when pressing [[Esc]] [#1734]
Expand Down
67 changes: 39 additions & 28 deletions src/internal/tabbable.ts
Original file line number Diff line number Diff line change
@@ -1,14 +1,22 @@
//
// This doesn't technically check visibility, it checks if the element has been rendered and can maybe possibly be tabbed
// to. This is a workaround for shadow roots not having an `offsetParent`.
//
// See https://stackoverflow.com/questions/19669786/check-if-element-is-visible-in-dom
//
// Previously, we used https://www.npmjs.com/package/composed-offset-position, but recursing up an entire node tree took
// up a lot of CPU cycles and made focus traps unusable in Chrome / Edge.
//
function isTakingUpSpace(elem: HTMLElement): boolean {
return Boolean(elem.offsetParent || elem.offsetWidth || elem.offsetHeight || elem.getClientRects().length);
// Cached compute style calls. This is specifically for browsers that dont support `checkVisibility()`.
// computedStyle calls are "live" so they only need to be retrieved once for an element.
const computedStyleMap = new WeakMap<Element, CSSStyleDeclaration>();

function isVisible(el: HTMLElement): boolean {
// This is the fastest check, but isn't supported in Safari.
if (typeof el.checkVisibility === 'function') {
return el.checkVisibility({ checkOpacity: false });
}

// Fallback "polyfill" for "checkVisibility"
let computedStyle: undefined | CSSStyleDeclaration = computedStyleMap.get(el);

if (!computedStyle) {
computedStyle = window.getComputedStyle(el, null);
computedStyleMap.set(el, computedStyle);
}

return computedStyle.visibility !== 'hidden' && computedStyle.display !== 'none';
}

/** Determines if the specified element is tabbable using heuristics inspired by https://github.com/focus-trap/tabbable */
Expand All @@ -30,13 +38,7 @@ function isTabbable(el: HTMLElement) {
return false;
}

// Elements that are hidden have no offsetParent and are not tabbable
if (!isTakingUpSpace(el)) {
return false;
}

// Elements without visibility are not tabbable
if (window.getComputedStyle(el).visibility === 'hidden') {
if (!isVisible(el)) {
return false;
}

Expand Down Expand Up @@ -73,7 +75,17 @@ export function getTabbableBoundary(root: HTMLElement | ShadowRoot) {
return { start, end };
}

/**
* This looks funky. Basically a slot's children will always be picked up *if* they're within the `root` element.
* However, there is an edge case when, if the `root` is wrapped by another shadow DOM, it won't grab the children.
* This fixes that fun edge case.
*/
function getSlottedChildrenOutsideRootElement(slotElement: HTMLSlotElement, root: HTMLElement | ShadowRoot) {
return (slotElement.getRootNode({ composed: true }) as ShadowRoot | null)?.host !== root;
}

export function getTabbableElements(root: HTMLElement | ShadowRoot) {
const walkedEls = new WeakMap();
const tabbableElements: HTMLElement[] = [];

function walk(el: HTMLElement | ShadowRoot) {
Expand All @@ -83,19 +95,16 @@ export function getTabbableElements(root: HTMLElement | ShadowRoot) {
return;
}

if (walkedEls.has(el)) {
return;
}
walkedEls.set(el, true);

if (!tabbableElements.includes(el) && isTabbable(el)) {
tabbableElements.push(el);
}

/**
* This looks funky. Basically a slot's children will always be picked up *if* they're within the `root` element.
* However, there is an edge case when, if the `root` is wrapped by another shadow DOM, it won't grab the children.
* This fixes that fun edge case.
*/
const slotChildrenOutsideRootElement = (slotElement: HTMLSlotElement) =>
(slotElement.getRootNode({ composed: true }) as ShadowRoot | null)?.host !== root;

if (el instanceof HTMLSlotElement && slotChildrenOutsideRootElement(el)) {
if (el instanceof HTMLSlotElement && getSlottedChildrenOutsideRootElement(el, root)) {
el.assignedElements({ flatten: true }).forEach((assignedEl: HTMLElement) => {
walk(assignedEl);
});
Expand All @@ -106,7 +115,9 @@ export function getTabbableElements(root: HTMLElement | ShadowRoot) {
}
}

[...el.children].forEach((e: HTMLElement) => walk(e));
for (const e of el.children) {
walk(e as HTMLElement);
}
}

// Collect all elements including the root
Expand Down

0 comments on commit dd27db5

Please sign in to comment.