Skip to content

Commit

Permalink
fix: handle more nested button group cases
Browse files Browse the repository at this point in the history
  • Loading branch information
aldeed committed Sep 8, 2020
1 parent d4e230f commit 25d5404
Show file tree
Hide file tree
Showing 9 changed files with 275 additions and 71 deletions.
45 changes: 44 additions & 1 deletion packages/sandbox/src/pages/Buttons/HtmlButtons.js
@@ -1,6 +1,15 @@
import React from 'react';

const buttonImage = '';

function HtmlButtons() {
const inputButtonsStyle = {
alignItems: 'end',
display: 'inline-flex',
justifyContent: 'space-between',
width: 380,
};

return (
<div className="container">
<h3>Native HTML</h3>
Expand Down Expand Up @@ -46,7 +55,41 @@ function HtmlButtons() {
</button>
<br />
<br />
<input id="submit-input" type="submit" value="Submit Input" />
<button type="button">
<svg xmlns="http://www.w3.org/2000/svg" width="10" height="10" style={{ marginRight: 6 }}>
<circle cx="5" cy="5" r="5" stroke="black" strokeWidth="0" fill="red"></circle>
</svg>
<span>SVG text button</span>
</button>
<br />
<br />
<button data-qa="nested-svg" type="button" style={{ display: 'inline-flex', alignItems: 'center' }}>
<svg xmlns="http://www.w3.org/2000/svg" width="10" height="10" style={{ marginRight: 6 }}>
<circle cx="5" cy="5" r="5" stroke="black" strokeWidth="0" fill="red"></circle>
</svg>
<div>
<span>Nested SVG text button</span>
</div>
</button>
<br />
<br />
<button data-qa="nested-svg-with-nested-link" type="button" style={{ display: 'inline-flex', alignItems: 'center' }}>
<svg xmlns="http://www.w3.org/2000/svg" width="10" height="10" style={{ marginRight: 6 }}>
<circle cx="5" cy="5" r="5" stroke="black" strokeWidth="0" fill="red"></circle>
</svg>
<div>
<span>Nested SVG text button&nbsp;</span>
</div>
<a href="#"><span>with nested link</span></a>
</button>
<br />
<br />
<div style={inputButtonsStyle}>
<input id="submit-input" type="submit" value="Submit Input" />
<input id="button-input" type="button" value="Button Input" />
<input id="image-input" type="image" src={buttonImage} alt="Play" width="30" />
<input id="reset-input" type="reset" value="Reset Input" />
</div>
<br />
<br />
<button data-qa="reload-top" onClick={() => window.top.location.reload()}>
Expand Down
13 changes: 5 additions & 8 deletions src/web/PageEventCollector.ts
@@ -1,7 +1,6 @@
import { DEFAULT_ATTRIBUTE_LIST } from './attribute';
import {
getInputElementValue,
getClickableGroup,
getTopmostEditableElement,
isVisible,
} from './element';
Expand Down Expand Up @@ -40,7 +39,11 @@ export class PageEventCollector {
});

this._onDispose.push(() =>
document.removeEventListener(eventName, handler),
document.removeEventListener(eventName, handler, {
// `capture` value must match for proper removal
// https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/removeEventListener#Matching_event_listeners_for_removal
capture: true,
}),
);
}

Expand All @@ -56,18 +59,12 @@ export class PageEventCollector {

const target = getTopmostEditableElement(event.target as HTMLElement);

let targetGroup: HTMLElement[];
if (isClick) {
targetGroup = getClickableGroup(target);
}

const isTargetVisible = isVisible(target, window.getComputedStyle(target));

const selector = buildSelector({
attributes: this._attributes,
isClick,
target,
targetGroup,
});

const elementEvent: types.ElementEvent = {
Expand Down
1 change: 0 additions & 1 deletion src/web/cues.ts
Expand Up @@ -13,7 +13,6 @@ export type BuildCues = {
attributes: string[];
isClick: boolean;
target: HTMLElement;
targetGroup?: HTMLElement[];
};

type CueTypeConfig = {
Expand Down
87 changes: 72 additions & 15 deletions src/web/element.ts
@@ -1,5 +1,8 @@
import { getXpath } from './serialize';

const BUTTON_INPUT_TYPES = ['button', 'image', 'reset', 'submit'];
const MAX_CLICKABLE_ELEMENT_TRAVERSE_DEPTH = 10;

export const isVisible = (
element: Element,
computedStyle?: CSSStyleDeclaration,
Expand All @@ -22,14 +25,79 @@ export const isVisible = (

export const isClickable = (
element: HTMLElement,
computedStyle: CSSStyleDeclaration,
computedStyle: CSSStyleDeclaration = window.getComputedStyle(element),
): boolean => {
// assume it is clickable if the cursor is a pointer
const clickable = computedStyle.cursor === 'pointer';

return clickable && isVisible(element, computedStyle);
};

export const isLikelyTopOfClickGroup = (element: HTMLElement): boolean => {
const tagName = element.tagName.toLowerCase();
return (
['a', 'button'].includes(tagName) ||
element.getAttribute('role') === 'button' ||
(tagName === 'input' && BUTTON_INPUT_TYPES.includes(element.getAttribute('type'))) ||
!element.parentElement ||
!isClickable(element.parentElement)
);
};

/**
* @summary Traverse the DOM in both directions, adding clickable elements to the array
* until we've determined the full set of elements that are in the clickable area.
* This is not foolproof because we can't know where exactly click handlers might
* be attached, but we can do a pretty good job of guessing.
*/
const traverseClickableElements = (
element: HTMLElement,
group: HTMLElement[],
direction: 'up' | 'down' = 'up',
maxDepth: number = MAX_CLICKABLE_ELEMENT_TRAVERSE_DEPTH,
depth = 0,
ancestorChain: string[] = [],
): void => {
// Regardless of which direction we're moving, stop if we hit a non-clickable element
if (!isClickable(element)) return;

// When moving up, when we reach the topmost clickable element, we
// stop traversing up and begin traversing down from there.
if (direction === 'up' && isLikelyTopOfClickGroup(element)) {
traverseClickableElements(element, group, 'down', maxDepth);
return;
}

// When moving down, stop if we hit the top of another click group (nested buttons)
if (direction === 'down' && depth > 0 && isLikelyTopOfClickGroup(element)) return;

const newDepth = depth + 1;
if (newDepth > maxDepth) return;

const lowerTagName = element.tagName.toLowerCase();

if (direction === 'up') {
// Call self for the parent element, incrementing depth
traverseClickableElements(element.parentElement, group, direction, maxDepth, newDepth, [lowerTagName, ...ancestorChain]);
} else {
// If we make it this far, this element should be part of the current group.
// We add elements to the group only on the way down to avoid adding any twice.
group.push(element);

const newAncestorChain = [...ancestorChain, lowerTagName];
console.debug('qawolf: added %s to click group', newAncestorChain.join(' > '));

// For now, let's skip going down into SVG descendants to keep this fast. If anyone
// finds a situation in which this causes problems, we can remove this.
if (lowerTagName !== 'svg') {
for (const child of element.children) {
// Call self for each child element, incrementing depth
traverseClickableElements(child as HTMLElement, group, direction, maxDepth, newDepth, newAncestorChain);
}
}
}
};

/**
* @summary Sometimes there is a group of elements that together make up what appears
* to be a single button, link, image, etc. Examples: a > span | button > div > span
Expand All @@ -45,21 +113,10 @@ export const getClickableGroup = (element: HTMLElement): HTMLElement[] => {
console.debug('qawolf: get clickable ancestor for', getXpath(element));

const clickableElements = [];
let checkElement = element;

while (isClickable(checkElement, window.getComputedStyle(checkElement))) {
clickableElements.push(checkElement);

if (['a', 'button', 'input'].includes(checkElement.tagName.toLowerCase())) {
// stop crawling when the checkElement is a good clickable tag
break;
}

checkElement = checkElement.parentElement;

// stop crawling at the root
if (!checkElement) break;
}
// Recursive function that will mutate clickableElements array. A recursive
// function is better than loops to avoid blocking UI paint.
traverseClickableElements(element, clickableElements);

return clickableElements;
};
Expand Down
58 changes: 28 additions & 30 deletions src/web/optimizeCues.ts
Expand Up @@ -7,7 +7,7 @@ import {
import { buildSelectorParts, isMatch } from './selectorEngine';
import { SelectorPart } from './types';

type CueGroup = {
export type CueGroup = {
cues: Cue[];
penalty: number;
selectorParts: SelectorPart[];
Expand Down Expand Up @@ -183,9 +183,8 @@ export const trimExcessCues = (
// Pick the cues that match the target with the lowest penalty
export const findBestCueGroup = (
seedGroup: CueGroup,
target: HTMLElement,
maxSize: number,
targetGroup?: HTMLElement[],
targetGroup: HTMLElement[],
): CueGroup => {
let bestGroup = seedGroup;

Expand Down Expand Up @@ -219,16 +218,9 @@ export const findBestCueGroup = (

const selectorParts = buildSelectorParts(cues);

// If these selector parts match the `target` element or any element in the
// target group, if there is one, then it's currently the best group.
if (
(
targetGroup &&
targetGroup.length &&
targetGroup.some((groupElement) => isMatch({ selectorParts, target: groupElement }))
) ||
isMatch({ selectorParts, target })
) {
// If these selector parts match any element that we are targeting,
// then it's currently the best group.
if (targetGroup.some((target) => isMatch({ selectorParts, target }))) {
bestGroup = {
cues,
penalty,
Expand All @@ -243,14 +235,12 @@ export const findBestCueGroup = (
};

export const optimizeCues = (
cues: Cue[],
cueSets: Cue[][],
target: HTMLElement,
targetGroup?: HTMLElement[],
): CueGroup | null => {
const cueSets = buildCueSets(cues);

targetGroup: HTMLElement[],
): CueGroup[] => {
// Only use the first 50 cue sets (there should never be this many, usually just ~2-3)
const cueGroups = cueSets
return cueSets
.slice(0, 50)
.map((cueSet) => {
// Trim down the cue group to 10 if possible
Expand All @@ -261,20 +251,28 @@ export const optimizeCues = (
// 16 cues, samples of 5 is ~7000 combinations which took ~100ms on my machine
if (!cueGroup || cueGroup.cues.length > 16) return null;

return findBestCueGroup(cueGroup, target, 5, targetGroup);
return findBestCueGroup(cueGroup, 5, targetGroup);
})
// Ignore invalid groups
.filter((a) => !!a)
// Rank by the total penalty then by value length
.sort((a, b) => {
if (a.penalty < b.penalty) return -1;
if (a.penalty > b.penalty) return 1;
.filter((a) => !!a);
};

if (a.valueLength < b.valueLength) return -1;
if (a.valueLength > b.valueLength) return 1;
export const pickBestCueGroup = (cueGroups: CueGroup[]): CueGroup | null => {
let bestCueGroup: CueGroup;

return 0;
});
if (cueGroups.length === 0) return null;

// Rank by the total penalty then by value length. This will take less
// time than .sort and will not mutate the cueGroups array.
for (const cueGroup of cueGroups) {
if (
!bestCueGroup ||
cueGroup.penalty < bestCueGroup.penalty ||
(cueGroup.penalty === bestCueGroup.penalty && cueGroup.valueLength < bestCueGroup.valueLength)
) {
bestCueGroup = cueGroup;
}
}

return cueGroups.length ? cueGroups[0] : null;
return bestCueGroup;
};
30 changes: 25 additions & 5 deletions src/web/selector.ts
@@ -1,5 +1,9 @@
import { buildCues, BuildCues } from './cues';
import { optimizeCues } from './optimizeCues';
import {
buildCues,
BuildCues,
} from './cues';
import { getClickableGroup } from './element';
import { buildCueSets, CueGroup, optimizeCues, pickBestCueGroup } from './optimizeCues';
import { getXpath } from './serialize';
import { isMatch } from './selectorEngine';
import { SelectorPart } from './types';
Expand Down Expand Up @@ -28,7 +32,15 @@ export const toSelector = (selectorParts: SelectorPart[]): string => {
};

export const buildSelector = (options: BuildCues): string => {
const { isClick, target, targetGroup } = options;
const { isClick, target } = options;

let targetGroup: HTMLElement[];
if (isClick) {
targetGroup = getClickableGroup(target);
if (targetGroup.length === 0) targetGroup = [target];
} else {
targetGroup = [target];
}

// To save looping, see if we have already figured out a unique
// selector for this target.
Expand All @@ -50,9 +62,17 @@ export const buildSelector = (options: BuildCues): string => {
}
}

const cues = buildCues(options);
const cueGroups: CueGroup[] = [];

targetGroup.forEach((targetElement) => {
const cues = buildCues({ ...options, target: targetElement });
const cueSets = buildCueSets(cues);
cueGroups.push(
...optimizeCues(cueSets, targetElement, targetGroup)
);
});

const { selectorParts } = optimizeCues(cues, target, targetGroup) || {};
const { selectorParts } = pickBestCueGroup(cueGroups) || {};
if (selectorParts) {
// First cache it so that we don't need to do all the looping for this
// same target next time. We cache `selectorParts` rather than `selector`
Expand Down

0 comments on commit 25d5404

Please sign in to comment.