Skip to content

Commit

Permalink
optimize mask/unmask rule matching rrweb-io#1097
Browse files Browse the repository at this point in the history
  • Loading branch information
mdellanoce committed Apr 6, 2023
1 parent 4dbfa74 commit 39c1af0
Showing 1 changed file with 50 additions and 67 deletions.
117 changes: 50 additions & 67 deletions packages/rrweb-snapshot/src/snapshot.ts
Original file line number Diff line number Diff line change
Expand Up @@ -287,87 +287,67 @@ export function _isBlockedElement(
return false;
}

function elementClassMatchesRegex(el: HTMLElement, regex: RegExp): boolean {
for (let eIndex = el.classList.length; eIndex--; ) {
const className = el.classList[eIndex];
if (regex.test(className)) {
return true;
}
}
return false;
}

export function classMatchesRegex(
node: Node | null,
regex: RegExp,
checkAncestors: boolean,
): boolean {
return distanceToClassRegexMatch(node, regex, checkAncestors) >= 0;
if (!node) return false;
if (checkAncestors) {
return (
distanceToMatch(node, (node) =>
elementClassMatchesRegex(node as HTMLElement, regex),
) >= 0
);
} else if (node.nodeType === node.ELEMENT_NODE) {
return elementClassMatchesRegex(node as HTMLElement, regex);
}
return false;
}

function distanceToClassRegexMatch(
function distanceToMatch(
node: Node | null,
regex: RegExp,
checkAncestors: boolean,
matchPredicate: (node: Node) => boolean,
limit = Infinity,
distance = 0,
): number {
if (!node) return -1;
if (node.nodeType !== node.ELEMENT_NODE) {
if (!checkAncestors) return -1;
return distanceToClassRegexMatch(node.parentNode, regex, checkAncestors);
}

for (let eIndex = (node as HTMLElement).classList.length; eIndex--; ) {
const className = (node as HTMLElement).classList[eIndex];
if (regex.test(className)) {
return distance;
}
}
if (!checkAncestors) return -1;
return distanceToClassRegexMatch(
node.parentNode,
regex,
checkAncestors,
distance + 1,
);
if (node.nodeType !== node.ELEMENT_NODE) return -1;
if (distance > limit) return -1;
if (matchPredicate(node)) return distance;
return distanceToMatch(node.parentNode, matchPredicate, limit, distance + 1);
}

function distanceToSelectorMatch(el: HTMLElement, selector: string): number {
if (!el) return -1;
if (el.matches(selector)) return 0;
const closestParent = el.closest(selector);
if (closestParent) {
let current = el;
let distance = 0;
while (current && current !== closestParent) {
current = current.parentNode as HTMLElement;
if (!current) {
return -1;
}
distance++;
}
return distance;
}
return -1;
}

function distanceToMatch(
el: HTMLElement,
function createMatchPredicate(
className: string | RegExp | null,
selector: string | null,
): number {
let classDistance = -1;
let selectorDistance = -1;

if (className) {
if (typeof className === 'string') {
classDistance = distanceToSelectorMatch(el, `.${className}`);
} else {
classDistance = distanceToClassRegexMatch(el, className, true);
): (node: Node) => boolean {
return (node: Node) => {
const el = node as HTMLElement;
if (el === null) return false;

if (className) {
if (typeof className === 'string') {
if (el.matches(`.${className}`)) return true;
} else if (elementClassMatchesRegex(el, className)) {
return true;
}
}
}

if (selector) {
selectorDistance = distanceToSelectorMatch(el, selector);
}
if (selector && el.matches(selector)) return true;

return selectorDistance >= 0
? classDistance >= 0
? Math.min(classDistance, selectorDistance)
: selectorDistance
: classDistance >= 0
? classDistance
: -1;
return false;
};
}

export function needMaskingText(
Expand All @@ -386,16 +366,19 @@ export function needMaskingText(

const unmaskDistance = distanceToMatch(
el,
unmaskTextClass,
unmaskTextSelector,
createMatchPredicate(unmaskTextClass, unmaskTextSelector),
);

let maskDistance = -1;
if (maskAllText && unmaskDistance < 0) {
return true;
}

maskDistance = distanceToMatch(el, maskTextClass, maskTextSelector);
maskDistance = distanceToMatch(
el,
createMatchPredicate(maskTextClass, maskTextSelector),
unmaskDistance >= 0 ? unmaskDistance : Infinity,
);

return maskDistance >= 0
? unmaskDistance >= 0
Expand Down

0 comments on commit 39c1af0

Please sign in to comment.