mirror of
https://github.com/gorhill/uBlock.git
synced 2026-03-11 09:04:36 +00:00
Related issue:
- https://github.com/uBlockOrigin/uBlock-issues/issues/1226
Related commit:
- 9eb455ab5e
In the previous commit, the element picker dialog was
isolated from the page content. This commit is to also
isolate the svg layers from the page content.
With this commit, there is no longer a need for an anonymous
iframe and the isolated world iframe is now directly
embedded in the page.
As a result, pages are now unable to interfere with any
of the element picker user interface. Pages can now only
see an iframe, but are unable to see the content of that
iframe. The styles applied to the iframe are from a user
stylesheet, so as to ensure pages can't override the
iframe's style properties set by uBO.
1308 lines
41 KiB
JavaScript
1308 lines
41 KiB
JavaScript
/*******************************************************************************
|
|
|
|
uBlock Origin - a browser extension to block requests.
|
|
Copyright (C) 2014-present Raymond Hill
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see {http://www.gnu.org/licenses/}.
|
|
|
|
Home: https://github.com/gorhill/uBlock
|
|
*/
|
|
|
|
/* global CSS */
|
|
|
|
'use strict';
|
|
|
|
/******************************************************************************/
|
|
/******************************************************************************/
|
|
|
|
(async ( ) => {
|
|
|
|
/******************************************************************************/
|
|
|
|
const epickerId = vAPI.randomToken();
|
|
let epickerConnectionId;
|
|
|
|
let pickerRoot = document.querySelector(`[${vAPI.sessionId}]`);
|
|
if ( pickerRoot !== null ) { return; }
|
|
|
|
let pickerBootArgs;
|
|
|
|
const netFilterCandidates = [];
|
|
const cosmeticFilterCandidates = [];
|
|
|
|
let targetElements = [];
|
|
let candidateElements = [];
|
|
let bestCandidateFilter = null;
|
|
|
|
const lastNetFilterSession = window.location.host + window.location.pathname;
|
|
let lastNetFilterHostname = '';
|
|
let lastNetFilterUnion = '';
|
|
|
|
/******************************************************************************/
|
|
|
|
const safeQuerySelectorAll = function(node, selector) {
|
|
if ( node !== null ) {
|
|
try {
|
|
return node.querySelectorAll(selector);
|
|
} catch (e) {
|
|
}
|
|
}
|
|
return [];
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const getElementBoundingClientRect = function(elem) {
|
|
let rect = typeof elem.getBoundingClientRect === 'function'
|
|
? elem.getBoundingClientRect()
|
|
: { height: 0, left: 0, top: 0, width: 0 };
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1024
|
|
// Try not returning an empty bounding rect.
|
|
if ( rect.width !== 0 && rect.height !== 0 ) {
|
|
return rect;
|
|
}
|
|
|
|
let left = rect.left,
|
|
right = rect.right,
|
|
top = rect.top,
|
|
bottom = rect.bottom;
|
|
|
|
for ( const child of elem.children ) {
|
|
rect = getElementBoundingClientRect(child);
|
|
if ( rect.width === 0 || rect.height === 0 ) {
|
|
continue;
|
|
}
|
|
if ( rect.left < left ) { left = rect.left; }
|
|
if ( rect.right > right ) { right = rect.right; }
|
|
if ( rect.top < top ) { top = rect.top; }
|
|
if ( rect.bottom > bottom ) { bottom = rect.bottom; }
|
|
}
|
|
|
|
return {
|
|
height: bottom - top,
|
|
left,
|
|
top,
|
|
width: right - left
|
|
};
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const highlightElements = function(elems, force) {
|
|
// To make mouse move handler more efficient
|
|
if (
|
|
(force !== true) &&
|
|
(elems.length === targetElements.length) &&
|
|
(elems.length === 0 || elems[0] === targetElements[0])
|
|
) {
|
|
return;
|
|
}
|
|
targetElements = [];
|
|
|
|
const ow = self.innerWidth;
|
|
const oh = self.innerHeight;
|
|
const islands = [];
|
|
|
|
for ( const elem of elems ) {
|
|
if ( elem === pickerRoot ) { continue; }
|
|
targetElements.push(elem);
|
|
const rect = getElementBoundingClientRect(elem);
|
|
// Ignore offscreen areas
|
|
if (
|
|
rect.left > ow || rect.top > oh ||
|
|
rect.left + rect.width < 0 || rect.top + rect.height < 0
|
|
) {
|
|
continue;
|
|
}
|
|
islands.push(
|
|
`M${rect.left} ${rect.top}h${rect.width}v${rect.height}h-${rect.width}z`
|
|
);
|
|
}
|
|
|
|
vAPI.MessagingConnection.sendTo(epickerConnectionId, {
|
|
what: 'svgPaths',
|
|
ocean: `M0 0h${ow}v${oh}h-${ow}z`,
|
|
islands: islands.join(''),
|
|
});
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const mergeStrings = function(urls) {
|
|
if ( urls.length === 0 ) { return ''; }
|
|
if (
|
|
urls.length === 1 ||
|
|
self.diff_match_patch instanceof Function === false
|
|
) {
|
|
return urls[0];
|
|
}
|
|
const differ = new self.diff_match_patch();
|
|
let merged = urls[0];
|
|
for ( let i = 1; i < urls.length; i++ ) {
|
|
// The differ works at line granularity: we insert a linefeed after
|
|
// each character to trick the differ to work at character granularity.
|
|
const diffs = differ.diff_main(
|
|
urls[i].split('').join('\n'),
|
|
merged.split('').join('\n')
|
|
);
|
|
const result = [];
|
|
for ( const diff of diffs ) {
|
|
if ( diff[0] !== 0 ) {
|
|
result.push('*');
|
|
} else {
|
|
result.push(diff[1].replace(/\n+/g, ''));
|
|
}
|
|
merged = result.join('');
|
|
}
|
|
}
|
|
// Keep usage of wildcards to a sane level, too many of them can cause
|
|
// high overhead filters
|
|
merged = merged.replace(/^\*+$/, '')
|
|
.replace(/\*{2,}/g, '*')
|
|
.replace(/([^*]{1,3}\*)(?:[^*]{1,3}\*)+/g, '$1');
|
|
return merged;
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// Remove fragment part from a URL.
|
|
|
|
const trimFragmentFromURL = function(url) {
|
|
const pos = url.indexOf('#');
|
|
return pos !== -1 ? url.slice(0, pos) : url;
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1897
|
|
// Ignore `data:` URI, they can't be handled by an HTTP observer.
|
|
|
|
const backgroundImageURLFromElement = function(elem) {
|
|
const style = window.getComputedStyle(elem);
|
|
const bgImg = style.backgroundImage || '';
|
|
const matches = /^url\((["']?)([^"']+)\1\)$/.exec(bgImg);
|
|
const url = matches !== null && matches.length === 3 ? matches[2] : '';
|
|
return url.lastIndexOf('data:', 0) === -1
|
|
? trimFragmentFromURL(url.slice(0, 1024))
|
|
: '';
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1725#issuecomment-226479197
|
|
// Limit returned string to 1024 characters.
|
|
// Also, return only URLs which will be seen by an HTTP observer.
|
|
|
|
const resourceURLsFromElement = function(elem) {
|
|
const urls = [];
|
|
const tagName = elem.localName;
|
|
const prop = netFilter1stSources[tagName];
|
|
if ( prop === undefined ) {
|
|
const url = backgroundImageURLFromElement(elem);
|
|
if ( url !== '' ) { urls.push(url); }
|
|
return urls;
|
|
}
|
|
{
|
|
const s = elem[prop];
|
|
if ( typeof s === 'string' && /^https?:\/\//.test(s) ) {
|
|
urls.push(trimFragmentFromURL(s.slice(0, 1024)));
|
|
}
|
|
}
|
|
resourceURLsFromSrcset(elem, urls);
|
|
return urls;
|
|
};
|
|
|
|
// https://html.spec.whatwg.org/multipage/images.html#parsing-a-srcset-attribute
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/1071
|
|
const resourceURLsFromSrcset = function(elem, out) {
|
|
let srcset = elem.srcset;
|
|
if ( typeof srcset !== 'string' || srcset === '' ) { return; }
|
|
for(;;) {
|
|
// trim whitespace
|
|
srcset = srcset.trim();
|
|
if ( srcset.length === 0 ) { break; }
|
|
// abort in case of leading comma
|
|
if ( /^,/.test(srcset) ) { break; }
|
|
// collect and consume all non-whitespace characters
|
|
let match = /^\S+/.exec(srcset);
|
|
if ( match === null ) { break; }
|
|
srcset = srcset.slice(match.index + match[0].length);
|
|
let url = match[0];
|
|
// consume descriptor, if any
|
|
if ( /,$/.test(url) ) {
|
|
url = url.replace(/,$/, '');
|
|
if ( /,$/.test(url) ) { break; }
|
|
} else {
|
|
match = /^[^,]*(?:\(.+?\))?[^,]*(?:,|$)/.exec(srcset);
|
|
if ( match === null ) { break; }
|
|
srcset = srcset.slice(match.index + match[0].length);
|
|
}
|
|
const parsedURL = new URL(url, document.baseURI);
|
|
if ( parsedURL.pathname.length === 0 ) { continue; }
|
|
out.push(trimFragmentFromURL(parsedURL.href));
|
|
}
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const netFilterFromUnion = function(patternIn, out) {
|
|
// Reset reference filter when dealing with unrelated URLs
|
|
const currentHostname = self.location.hostname;
|
|
if (
|
|
lastNetFilterUnion === '' ||
|
|
currentHostname === '' ||
|
|
currentHostname !== lastNetFilterHostname
|
|
) {
|
|
lastNetFilterHostname = currentHostname;
|
|
lastNetFilterUnion = patternIn;
|
|
vAPI.messaging.send('elementPicker', {
|
|
what: 'elementPickerEprom',
|
|
lastNetFilterSession,
|
|
lastNetFilterHostname,
|
|
lastNetFilterUnion,
|
|
});
|
|
return;
|
|
}
|
|
|
|
// Related URLs
|
|
lastNetFilterHostname = currentHostname;
|
|
let patternOut = mergeStrings([ patternIn, lastNetFilterUnion ]);
|
|
if ( patternOut !== '/*' && patternOut !== patternIn ) {
|
|
const filter = `||${patternOut}`;
|
|
if ( out.indexOf(filter) === -1 ) {
|
|
out.push(filter);
|
|
}
|
|
lastNetFilterUnion = patternOut;
|
|
}
|
|
|
|
// Remember across element picker sessions
|
|
vAPI.messaging.send('elementPicker', {
|
|
what: 'elementPickerEprom',
|
|
lastNetFilterSession,
|
|
lastNetFilterHostname,
|
|
lastNetFilterUnion,
|
|
});
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// Extract the best possible net filter, i.e. as specific as possible.
|
|
|
|
const netFilterFromElement = function(elem) {
|
|
if ( elem === null ) { return 0; }
|
|
if ( elem.nodeType !== 1 ) { return 0; }
|
|
const urls = resourceURLsFromElement(elem);
|
|
if ( urls.length === 0 ) { return 0; }
|
|
|
|
if ( candidateElements.indexOf(elem) === -1 ) {
|
|
candidateElements.push(elem);
|
|
}
|
|
|
|
const candidates = netFilterCandidates;
|
|
const len = candidates.length;
|
|
|
|
for ( let i = 0; i < urls.length; i++ ) {
|
|
urls[i] = urls[i].replace(/^https?:\/\//, '');
|
|
}
|
|
const pattern = mergeStrings(urls);
|
|
|
|
|
|
if ( bestCandidateFilter === null ) {
|
|
bestCandidateFilter = {
|
|
type: 'net',
|
|
filters: candidates,
|
|
slot: candidates.length
|
|
};
|
|
}
|
|
|
|
candidates.push(`||${pattern}`);
|
|
|
|
// Suggest a less narrow filter if possible
|
|
const pos = pattern.indexOf('?');
|
|
if ( pos !== -1 ) {
|
|
candidates.push(`||${pattern.slice(0, pos)}`);
|
|
}
|
|
|
|
// Suggest a filter which is a result of combining more than one URL.
|
|
netFilterFromUnion(pattern, candidates);
|
|
|
|
return candidates.length - len;
|
|
};
|
|
|
|
const netFilter1stSources = {
|
|
'audio': 'src',
|
|
'embed': 'src',
|
|
'iframe': 'src',
|
|
'img': 'src',
|
|
'object': 'data',
|
|
'video': 'src'
|
|
};
|
|
|
|
const filterTypes = {
|
|
'audio': 'media',
|
|
'embed': 'object',
|
|
'iframe': 'subdocument',
|
|
'img': 'image',
|
|
'object': 'object',
|
|
'video': 'media',
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// Extract the best possible cosmetic filter, i.e. as specific as possible.
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1725
|
|
// Also take into account the `src` attribute for `img` elements -- and limit
|
|
// the value to the 1024 first characters.
|
|
|
|
const cosmeticFilterFromElement = function(elem) {
|
|
if ( elem === null ) { return 0; }
|
|
if ( elem.nodeType !== 1 ) { return 0; }
|
|
|
|
if ( candidateElements.indexOf(elem) === -1 ) {
|
|
candidateElements.push(elem);
|
|
}
|
|
|
|
let selector = '';
|
|
|
|
// Id
|
|
let v = typeof elem.id === 'string' && CSS.escape(elem.id);
|
|
if ( v ) {
|
|
selector = '#' + v;
|
|
}
|
|
|
|
// Class(es)
|
|
v = elem.classList;
|
|
if ( v ) {
|
|
let i = v.length || 0;
|
|
while ( i-- ) {
|
|
selector += '.' + CSS.escape(v.item(i));
|
|
}
|
|
}
|
|
|
|
// Tag name
|
|
const tagName = elem.localName;
|
|
|
|
// Use attributes if still no selector found.
|
|
// https://github.com/gorhill/uBlock/issues/1901
|
|
// Trim attribute value, this may help in case of malformed HTML.
|
|
if ( selector === '' ) {
|
|
let attributes = [], attr;
|
|
switch ( tagName ) {
|
|
case 'a':
|
|
v = elem.getAttribute('href');
|
|
if ( v ) {
|
|
v = v.trim().replace(/\?.*$/, '');
|
|
if ( v.length ) {
|
|
attributes.push({ k: 'href', v: v });
|
|
}
|
|
}
|
|
break;
|
|
case 'iframe':
|
|
case 'img':
|
|
v = elem.getAttribute('src');
|
|
if ( v && v.length !== 0 ) {
|
|
v = v.trim();
|
|
if ( v.startsWith('data:') ) {
|
|
let pos = v.indexOf(',');
|
|
if ( pos !== -1 ) {
|
|
v = v.slice(0, pos + 1);
|
|
}
|
|
} else if ( v.startsWith('blob:') ) {
|
|
v = new URL(v.slice(5));
|
|
v.pathname = '';
|
|
v = 'blob:' + v.href;
|
|
}
|
|
attributes.push({ k: 'src', v: v.slice(0, 256) });
|
|
break;
|
|
}
|
|
v = elem.getAttribute('alt');
|
|
if ( v && v.length !== 0 ) {
|
|
attributes.push({ k: 'alt', v: v });
|
|
break;
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
while ( (attr = attributes.pop()) ) {
|
|
if ( attr.v.length === 0 ) { continue; }
|
|
v = elem.getAttribute(attr.k);
|
|
if ( attr.v === v ) {
|
|
selector += `[${attr.k}="${attr.v}"]`;
|
|
} else if ( v.startsWith(attr.v) ) {
|
|
selector += `[${attr.k}^="${attr.v}"]`;
|
|
} else {
|
|
selector += `[${attr.k}*="${attr.v}"]`;
|
|
}
|
|
}
|
|
}
|
|
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/17
|
|
// If selector is ambiguous at this point, add the element name to
|
|
// further narrow it down.
|
|
const parentNode = elem.parentNode;
|
|
if (
|
|
selector === '' ||
|
|
safeQuerySelectorAll(parentNode, `:scope > ${selector}`).length > 1
|
|
) {
|
|
selector = tagName + selector;
|
|
}
|
|
|
|
// https://github.com/chrisaljoudi/uBlock/issues/637
|
|
// If the selector is still ambiguous at this point, further narrow using
|
|
// `nth-of-type`. It is preferable to use `nth-of-type` as opposed to
|
|
// `nth-child`, as `nth-of-type` is less volatile.
|
|
if ( safeQuerySelectorAll(parentNode, `:scope > ${selector}`).length > 1 ) {
|
|
let i = 1;
|
|
while ( elem.previousSibling !== null ) {
|
|
elem = elem.previousSibling;
|
|
if (
|
|
typeof elem.localName === 'string' &&
|
|
elem.localName === tagName
|
|
) {
|
|
i++;
|
|
}
|
|
}
|
|
selector += `:nth-of-type(${i})`;
|
|
}
|
|
|
|
if ( bestCandidateFilter === null ) {
|
|
bestCandidateFilter = {
|
|
type: 'cosmetic',
|
|
filters: cosmeticFilterCandidates,
|
|
slot: cosmeticFilterCandidates.length
|
|
};
|
|
}
|
|
|
|
cosmeticFilterCandidates.push(`##${selector}`);
|
|
|
|
return 1;
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const filtersFrom = function(x, y) {
|
|
bestCandidateFilter = null;
|
|
netFilterCandidates.length = 0;
|
|
cosmeticFilterCandidates.length = 0;
|
|
candidateElements.length = 0;
|
|
|
|
// We need at least one element.
|
|
let first = null;
|
|
if ( typeof x === 'number' ) {
|
|
first = elementFromPoint(x, y);
|
|
} else if ( x instanceof HTMLElement ) {
|
|
first = x;
|
|
x = undefined;
|
|
}
|
|
|
|
// Network filter from element which was clicked.
|
|
if ( first !== null ) {
|
|
netFilterFromElement(first);
|
|
}
|
|
|
|
// Cosmetic filter candidates from ancestors.
|
|
let elem = first;
|
|
while ( elem && elem !== document.body ) {
|
|
cosmeticFilterFromElement(elem);
|
|
elem = elem.parentNode;
|
|
}
|
|
// The body tag is needed as anchor only when the immediate child
|
|
// uses `nth-of-type`.
|
|
let i = cosmeticFilterCandidates.length;
|
|
if ( i !== 0 ) {
|
|
let selector = cosmeticFilterCandidates[i-1];
|
|
if (
|
|
selector.indexOf(':nth-of-type(') !== -1 &&
|
|
safeQuerySelectorAll(document.body, selector).length > 1
|
|
) {
|
|
cosmeticFilterCandidates.push('##body');
|
|
}
|
|
}
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1545
|
|
// Network filter candidates from all other elements found at
|
|
// point (x, y).
|
|
if ( typeof x === 'number' ) {
|
|
const attrName = vAPI.sessionId + '-clickblind';
|
|
elem = first;
|
|
while ( elem !== null ) {
|
|
const previous = elem;
|
|
elem.setAttribute(attrName, '');
|
|
elem = elementFromPoint(x, y);
|
|
if ( elem === null || elem === previous ) { break; }
|
|
netFilterFromElement(elem);
|
|
}
|
|
for ( const elem of document.querySelectorAll(`[${attrName}]`) ) {
|
|
elem.removeAttribute(attrName);
|
|
}
|
|
|
|
netFilterFromElement(document.body);
|
|
}
|
|
|
|
return netFilterCandidates.length + cosmeticFilterCandidates.length;
|
|
};
|
|
|
|
/*******************************************************************************
|
|
|
|
filterToDOMInterface.queryAll
|
|
@desc Look-up all the HTML elements matching the filter passed in
|
|
argument.
|
|
@param string, a cosmetic or network filter.
|
|
@param function, called once all items matching the filter have been
|
|
collected.
|
|
@return array, or undefined if the filter is invalid.
|
|
|
|
filterToDOMInterface.preview
|
|
@desc Apply/unapply filter to the DOM.
|
|
@param string, a cosmetic of network filter, or literal false to remove
|
|
the effects of the filter on the DOM.
|
|
@return undefined.
|
|
|
|
TODO: need to be revised once I implement chained cosmetic operators.
|
|
|
|
*/
|
|
|
|
const filterToDOMInterface = (( ) => {
|
|
const reHnAnchorPrefix = '^[\\w-]+://(?:[^/?#]+\\.)?';
|
|
const reCaret = '(?:[^%.0-9a-z_-]|$)';
|
|
const rePseudoElements = /:(?::?after|:?before|:[a-z-]+)$/;
|
|
|
|
// Net filters: we need to lookup manually -- translating into a foolproof
|
|
// CSS selector is just not possible.
|
|
//
|
|
// https://github.com/chrisaljoudi/uBlock/issues/945
|
|
// Transform into a regular expression, this allows the user to
|
|
// edit and insert wildcard(s) into the proposed filter.
|
|
// https://www.reddit.com/r/uBlockOrigin/comments/c5do7w/
|
|
// Better handling of pure hostname filters. Also, discard single
|
|
// alphanumeric character filters.
|
|
const fromNetworkFilter = function(filter) {
|
|
const out = [];
|
|
if ( /^[0-9a-z]$/i.test(filter) ) { return out; }
|
|
let reStr = '';
|
|
if (
|
|
filter.length > 2 &&
|
|
filter.startsWith('/') &&
|
|
filter.endsWith('/')
|
|
) {
|
|
reStr = filter.slice(1, -1);
|
|
} else if ( /^\w[\w.-]*[a-z]$/i.test(filter) ) {
|
|
reStr = reHnAnchorPrefix +
|
|
filter.toLowerCase().replace(/\./g, '\\.') +
|
|
reCaret;
|
|
} else {
|
|
let rePrefix = '', reSuffix = '';
|
|
if ( filter.startsWith('||') ) {
|
|
rePrefix = reHnAnchorPrefix;
|
|
filter = filter.slice(2);
|
|
} else if ( filter.startsWith('|') ) {
|
|
rePrefix = '^';
|
|
filter = filter.slice(1);
|
|
}
|
|
if ( filter.endsWith('|') ) {
|
|
reSuffix = '$';
|
|
filter = filter.slice(0, -1);
|
|
}
|
|
reStr = rePrefix +
|
|
filter.replace(/[.+?${}()|[\]\\]/g, '\\$&')
|
|
.replace(/\*+/g, '.*')
|
|
.replace(/\^/g, reCaret) +
|
|
reSuffix;
|
|
}
|
|
let reFilter = null;
|
|
try {
|
|
reFilter = new RegExp(reStr, 'i');
|
|
}
|
|
catch (e) {
|
|
return out;
|
|
}
|
|
|
|
// Lookup by tag names.
|
|
const elems = document.querySelectorAll(
|
|
Object.keys(netFilter1stSources).join()
|
|
);
|
|
for ( const elem of elems ) {
|
|
const srcProp = netFilter1stSources[elem.localName];
|
|
const src = elem[srcProp];
|
|
if (
|
|
typeof src === 'string' &&
|
|
reFilter.test(src) ||
|
|
typeof elem.currentSrc === 'string' &&
|
|
reFilter.test(elem.currentSrc)
|
|
) {
|
|
out.push({
|
|
type: 'network',
|
|
elem: elem,
|
|
src: srcProp,
|
|
opts: filterTypes[elem.localName],
|
|
});
|
|
}
|
|
}
|
|
|
|
// Find matching background image in current set of candidate elements.
|
|
for ( const elem of candidateElements ) {
|
|
if ( reFilter.test(backgroundImageURLFromElement(elem)) ) {
|
|
out.push({
|
|
type: 'network',
|
|
elem: elem,
|
|
style: 'background-image',
|
|
opts: 'image',
|
|
});
|
|
}
|
|
}
|
|
|
|
return out;
|
|
};
|
|
|
|
// Cosmetic filters: these are straight CSS selectors.
|
|
//
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/389
|
|
// Test filter using comma-separated list to better detect invalid CSS
|
|
// selectors.
|
|
//
|
|
// https://github.com/gorhill/uBlock/issues/2515
|
|
// Remove trailing pseudo-element when querying.
|
|
const fromPlainCosmeticFilter = function(raw) {
|
|
let elems;
|
|
try {
|
|
document.documentElement.matches(`${raw},\na`);
|
|
elems = document.querySelectorAll(
|
|
raw.replace(rePseudoElements, '')
|
|
);
|
|
}
|
|
catch (e) {
|
|
return;
|
|
}
|
|
const out = [];
|
|
for ( const elem of elems ) {
|
|
if ( elem === pickerRoot ) { continue; }
|
|
out.push({ type: 'cosmetic', elem, raw });
|
|
}
|
|
return out;
|
|
};
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1772
|
|
// Handle procedural cosmetic filters.
|
|
//
|
|
// https://github.com/gorhill/uBlock/issues/2515
|
|
// Remove trailing pseudo-element when querying.
|
|
const fromCompiledCosmeticFilter = function(raw) {
|
|
if ( typeof raw !== 'string' ) { return; }
|
|
let elems;
|
|
try {
|
|
const o = JSON.parse(raw);
|
|
if ( o.action === 'style' ) {
|
|
elems = document.querySelectorAll(
|
|
o.selector.replace(rePseudoElements, '')
|
|
);
|
|
lastAction = o.selector + ' {' + o.tasks[0][1] + '}';
|
|
} else if ( o.tasks ) {
|
|
elems = vAPI.domFilterer.createProceduralFilter(o).exec();
|
|
}
|
|
} catch(ex) {
|
|
return;
|
|
}
|
|
if ( !elems ) { return; }
|
|
const out = [];
|
|
for ( const elem of elems ) {
|
|
out.push({ type: 'cosmetic', elem, raw });
|
|
}
|
|
return out;
|
|
};
|
|
|
|
let lastFilter;
|
|
let lastResultset;
|
|
let lastAction;
|
|
let appliedStyleTag;
|
|
let applied = false;
|
|
let previewing = false;
|
|
|
|
const queryAll = function(details) {
|
|
let { filter, compiled } = details;
|
|
filter = filter.trim();
|
|
if ( filter === lastFilter ) { return lastResultset; }
|
|
unapply();
|
|
if ( filter === '' || filter === '!' ) {
|
|
lastFilter = '';
|
|
lastResultset = [];
|
|
return lastResultset;
|
|
}
|
|
lastFilter = filter;
|
|
lastAction = undefined;
|
|
if ( filter.startsWith('##') === false ) {
|
|
lastResultset = fromNetworkFilter(filter);
|
|
if ( previewing ) { apply(); }
|
|
return lastResultset;
|
|
}
|
|
lastResultset = fromPlainCosmeticFilter(compiled);
|
|
if ( lastResultset ) {
|
|
if ( previewing ) { apply(); }
|
|
return lastResultset;
|
|
}
|
|
// Procedural cosmetic filter
|
|
lastResultset = fromCompiledCosmeticFilter(compiled);
|
|
if ( previewing ) { apply(); }
|
|
return lastResultset;
|
|
};
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1629
|
|
// Avoid hiding the element picker's related elements.
|
|
const applyHide = function() {
|
|
const htmlElem = document.documentElement;
|
|
for ( const item of lastResultset ) {
|
|
const elem = item.elem;
|
|
if ( elem === pickerRoot ) { continue; }
|
|
if (
|
|
(elem !== htmlElem) &&
|
|
(item.type === 'cosmetic' || item.type === 'network' && item.src !== undefined)
|
|
) {
|
|
vAPI.domFilterer.hideNode(elem);
|
|
item.hidden = true;
|
|
}
|
|
if ( item.type === 'network' && item.style === 'background-image' ) {
|
|
const style = elem.style;
|
|
item.backgroundImage = style.getPropertyValue('background-image');
|
|
item.backgroundImagePriority = style.getPropertyPriority('background-image');
|
|
style.setProperty('background-image', 'none', 'important');
|
|
}
|
|
}
|
|
};
|
|
|
|
const unapplyHide = function() {
|
|
if ( lastResultset === undefined ) { return; }
|
|
for ( const item of lastResultset ) {
|
|
if ( item.hidden === true ) {
|
|
vAPI.domFilterer.unhideNode(item.elem);
|
|
item.hidden = false;
|
|
}
|
|
if ( item.hasOwnProperty('backgroundImage') ) {
|
|
item.elem.style.setProperty(
|
|
'background-image',
|
|
item.backgroundImage,
|
|
item.backgroundImagePriority
|
|
);
|
|
delete item.backgroundImage;
|
|
}
|
|
}
|
|
};
|
|
|
|
const unapplyStyle = function() {
|
|
if ( !appliedStyleTag || appliedStyleTag.parentNode === null ) {
|
|
return;
|
|
}
|
|
appliedStyleTag.parentNode.removeChild(appliedStyleTag);
|
|
};
|
|
|
|
const applyStyle = function() {
|
|
if ( !appliedStyleTag ) {
|
|
appliedStyleTag = document.createElement('style');
|
|
appliedStyleTag.setAttribute('type', 'text/css');
|
|
}
|
|
appliedStyleTag.textContent = lastAction;
|
|
if ( appliedStyleTag.parentNode === null ) {
|
|
document.head.appendChild(appliedStyleTag);
|
|
}
|
|
};
|
|
|
|
const apply = function() {
|
|
if ( applied ) {
|
|
unapply();
|
|
}
|
|
if ( lastResultset === undefined ) { return; }
|
|
if ( typeof lastAction === 'string' ) {
|
|
applyStyle();
|
|
} else {
|
|
applyHide();
|
|
}
|
|
applied = true;
|
|
};
|
|
|
|
const unapply = function() {
|
|
if ( !applied ) { return; }
|
|
if ( typeof lastAction === 'string' ) {
|
|
unapplyStyle();
|
|
} else {
|
|
unapplyHide();
|
|
}
|
|
applied = false;
|
|
};
|
|
|
|
// https://www.reddit.com/r/uBlockOrigin/comments/c62irc/
|
|
// Support injecting the cosmetic filters into the DOM filterer
|
|
// immediately rather than wait for the next page load.
|
|
const preview = function(state, permanent = false) {
|
|
previewing = state !== false;
|
|
if ( previewing === false ) {
|
|
return unapply();
|
|
}
|
|
if ( lastResultset === undefined ) { return; }
|
|
apply();
|
|
if ( permanent === false ) { return; }
|
|
if ( vAPI.domFilterer instanceof Object === false ) { return; }
|
|
const cssSelectors = new Set();
|
|
const proceduralSelectors = new Set();
|
|
for ( const item of lastResultset ) {
|
|
if ( item.type !== 'cosmetic' ) { continue; }
|
|
if ( item.raw.startsWith('{') ) {
|
|
proceduralSelectors.add(item.raw);
|
|
} else {
|
|
cssSelectors.add(item.raw);
|
|
}
|
|
}
|
|
if ( cssSelectors.size !== 0 ) {
|
|
vAPI.domFilterer.addCSSRule(
|
|
Array.from(cssSelectors),
|
|
'display:none!important;'
|
|
);
|
|
}
|
|
if ( proceduralSelectors.size !== 0 ) {
|
|
vAPI.domFilterer.addProceduralSelectors(
|
|
Array.from(proceduralSelectors)
|
|
);
|
|
}
|
|
};
|
|
|
|
return {
|
|
get previewing() { return previewing; },
|
|
preview,
|
|
queryAll,
|
|
};
|
|
})();
|
|
|
|
/******************************************************************************/
|
|
|
|
const showDialog = function(options) {
|
|
vAPI.MessagingConnection.sendTo(epickerConnectionId, {
|
|
what: 'showDialog',
|
|
hostname: self.location.hostname,
|
|
origin: self.location.origin,
|
|
netFilters: netFilterCandidates,
|
|
cosmeticFilters: cosmeticFilterCandidates,
|
|
filter: bestCandidateFilter,
|
|
options,
|
|
});
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const elementFromPoint = (( ) => {
|
|
let lastX, lastY;
|
|
|
|
return (x, y) => {
|
|
if ( x !== undefined ) {
|
|
lastX = x; lastY = y;
|
|
} else if ( lastX !== undefined ) {
|
|
x = lastX; y = lastY;
|
|
} else {
|
|
return null;
|
|
}
|
|
if ( !pickerRoot ) { return null; }
|
|
const magicAttr = `${vAPI.sessionId}-clickblind`;
|
|
pickerRoot.setAttribute(magicAttr, '');
|
|
let elem = document.elementFromPoint(x, y);
|
|
if ( elem === document.body || elem === document.documentElement ) {
|
|
elem = null;
|
|
}
|
|
// https://github.com/uBlockOrigin/uBlock-issues/issues/380
|
|
pickerRoot.removeAttribute(magicAttr);
|
|
return elem;
|
|
};
|
|
})();
|
|
|
|
/******************************************************************************/
|
|
|
|
const highlightElementAtPoint = function(mx, my) {
|
|
const elem = elementFromPoint(mx, my);
|
|
highlightElements(elem ? [ elem ] : []);
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const filterElementAtPoint = function(mx, my, broad) {
|
|
if ( filtersFrom(mx, my) === 0 ) { return; }
|
|
showDialog({ broad });
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// https://www.reddit.com/r/uBlockOrigin/comments/bktxtb/scrolling_doesnt_work/emn901o
|
|
// Override 'fixed' position property on body element if present.
|
|
|
|
// With touch-driven devices, first highlight the element and remove only
|
|
// when tapping again the highlighted area.
|
|
|
|
const zapElementAtPoint = function(mx, my, options) {
|
|
if ( options.highlight ) {
|
|
const elem = elementFromPoint(mx, my);
|
|
if ( elem ) {
|
|
highlightElements([ elem ]);
|
|
}
|
|
return;
|
|
}
|
|
|
|
let elem = targetElements.length !== 0 && targetElements[0] || null;
|
|
if ( elem === null && mx !== undefined ) {
|
|
elem = elementFromPoint(mx, my);
|
|
}
|
|
|
|
if ( elem instanceof HTMLElement === false ) { return; }
|
|
|
|
const getStyleValue = function(elem, prop) {
|
|
const style = window.getComputedStyle(elem);
|
|
return style ? style[prop] : '';
|
|
};
|
|
|
|
// Heuristic to detect scroll-locking: remove such lock when detected.
|
|
if (
|
|
parseInt(getStyleValue(elem, 'zIndex'), 10) >= 1000 ||
|
|
getStyleValue(elem, 'position') === 'fixed'
|
|
) {
|
|
const doc = document;
|
|
if ( getStyleValue(doc.body, 'overflowY') === 'hidden' ) {
|
|
doc.body.style.setProperty('overflow', 'auto', 'important');
|
|
}
|
|
if ( getStyleValue(doc.body, 'position') === 'fixed' ) {
|
|
doc.body.style.setProperty('position', 'static', 'important');
|
|
}
|
|
if ( getStyleValue(doc.documentElement, 'overflowY') === 'hidden' ) {
|
|
doc.documentElement.style.setProperty('overflow', 'auto', 'important');
|
|
}
|
|
}
|
|
|
|
elem.remove();
|
|
highlightElementAtPoint(mx, my);
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const onKeyPressed = function(ev) {
|
|
// Delete
|
|
if (
|
|
(ev.key === 'Delete' || ev.key === 'Backspace') &&
|
|
pickerBootArgs.zap
|
|
) {
|
|
ev.stopPropagation();
|
|
ev.preventDefault();
|
|
zapElementAtPoint();
|
|
return;
|
|
}
|
|
// Esc
|
|
if ( ev.key === 'Escape' || ev.which === 27 ) {
|
|
ev.stopPropagation();
|
|
ev.preventDefault();
|
|
filterToDOMInterface.preview(false);
|
|
quitPicker();
|
|
return;
|
|
}
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// https://github.com/chrisaljoudi/uBlock/issues/190
|
|
// May need to dynamically adjust the height of the overlay + new position
|
|
// of highlighted elements.
|
|
|
|
const onViewportChanged = function() {
|
|
highlightElements(targetElements, true);
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// Auto-select a specific target, if any, and if possible
|
|
|
|
const startPicker = function() {
|
|
self.addEventListener('scroll', onViewportChanged, { passive: true });
|
|
self.addEventListener('resize', onViewportChanged, { passive: true });
|
|
self.addEventListener('keydown', onKeyPressed, true);
|
|
|
|
// Try using mouse position
|
|
if (
|
|
pickerBootArgs.mouse &&
|
|
typeof vAPI.mouseClick.x === 'number' &&
|
|
vAPI.mouseClick.x > 0
|
|
) {
|
|
if ( filtersFrom(vAPI.mouseClick.x, vAPI.mouseClick.y) !== 0 ) {
|
|
return showDialog();
|
|
}
|
|
}
|
|
|
|
// No mouse position available, use suggested target
|
|
const target = pickerBootArgs.target || '';
|
|
const pos = target.indexOf('\t');
|
|
if ( pos === -1 ) { return; }
|
|
|
|
const srcAttrMap = {
|
|
'a': 'href',
|
|
'audio': 'src',
|
|
'embed': 'src',
|
|
'iframe': 'src',
|
|
'img': 'src',
|
|
'video': 'src',
|
|
};
|
|
const tagName = target.slice(0, pos);
|
|
const url = target.slice(pos + 1);
|
|
const attr = srcAttrMap[tagName];
|
|
if ( attr === undefined ) { return; }
|
|
const elems = document.getElementsByTagName(tagName);
|
|
for ( const elem of elems ) {
|
|
if ( elem === pickerRoot ) { continue; }
|
|
const src = elem[attr];
|
|
if ( typeof src !== 'string' ) { continue; }
|
|
if ( (src !== url) && (src !== '' || url !== 'about:blank') ) {
|
|
continue;
|
|
}
|
|
elem.scrollIntoView({ behavior: 'smooth', block: 'start' });
|
|
filtersFrom(elem);
|
|
return showDialog({ broad: true });
|
|
}
|
|
|
|
// A target was specified, but it wasn't found: abort.
|
|
quitPicker();
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// Let's have the element picker code flushed from memory when no longer
|
|
// in use: to ensure this, release all local references.
|
|
|
|
const quitPicker = function() {
|
|
self.removeEventListener('scroll', onViewportChanged, { passive: true });
|
|
self.removeEventListener('resize', onViewportChanged, { passive: true });
|
|
self.removeEventListener('keydown', onKeyPressed, true);
|
|
vAPI.shutdown.remove(quitPicker);
|
|
vAPI.MessagingConnection.disconnectFrom(epickerConnectionId);
|
|
vAPI.MessagingConnection.removeListener(onConnectionMessage);
|
|
vAPI.userStylesheet.remove(pickerCSS);
|
|
vAPI.userStylesheet.apply();
|
|
|
|
if ( pickerRoot === null ) { return; }
|
|
|
|
// https://github.com/gorhill/uBlock/issues/2060
|
|
if ( vAPI.domFilterer instanceof Object ) {
|
|
vAPI.domFilterer.unexcludeNode(pickerRoot);
|
|
}
|
|
|
|
pickerRoot.remove();
|
|
pickerRoot = null;
|
|
|
|
self.focus();
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const onDialogMessage = function(msg) {
|
|
switch ( msg.what ) {
|
|
case 'start':
|
|
startPicker();
|
|
if ( targetElements.length === 0 ) {
|
|
highlightElements([], true);
|
|
}
|
|
break;
|
|
case 'dialogCreate':
|
|
filterToDOMInterface.queryAll(msg);
|
|
filterToDOMInterface.preview(true, true);
|
|
quitPicker();
|
|
break;
|
|
case 'dialogSetFilter': {
|
|
const resultset = filterToDOMInterface.queryAll(msg);
|
|
highlightElements(resultset.map(a => a.elem), true);
|
|
if ( msg.filter === '!' ) { break; }
|
|
vAPI.MessagingConnection.sendTo(epickerConnectionId, {
|
|
what: 'filterResultset',
|
|
resultset: resultset.map(a => {
|
|
const o = Object.assign({}, a);
|
|
o.elem = undefined;
|
|
return o;
|
|
}),
|
|
});
|
|
break;
|
|
}
|
|
case 'quitPicker':
|
|
filterToDOMInterface.preview(false);
|
|
quitPicker();
|
|
break;
|
|
case 'highlightElementAtPoint':
|
|
highlightElementAtPoint(msg.mx, msg.my);
|
|
break;
|
|
case 'unhighlight':
|
|
highlightElements([]);
|
|
break;
|
|
case 'filterElementAtPoint':
|
|
filterElementAtPoint(msg.mx, msg.my, msg.broad);
|
|
break;
|
|
case 'zapElementAtPoint':
|
|
zapElementAtPoint(msg.mx, msg.my, msg.options);
|
|
if ( msg.options.highlight !== true && msg.options.stay !== true ) {
|
|
quitPicker();
|
|
}
|
|
break;
|
|
case 'togglePreview':
|
|
filterToDOMInterface.preview(msg.state);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
const onConnectionMessage = function(msg) {
|
|
if ( msg.from !== `epickerDialog-${epickerId}` ) { return; }
|
|
switch ( msg.what ) {
|
|
case 'connectionRequested':
|
|
epickerConnectionId = msg.id;
|
|
return true;
|
|
case 'connectionBroken':
|
|
quitPicker();
|
|
break;
|
|
case 'connectionMessage':
|
|
onDialogMessage(msg.payload);
|
|
break;
|
|
}
|
|
};
|
|
|
|
/******************************************************************************/
|
|
|
|
// epicker-ui.html will be injected in the page through an iframe, and
|
|
// is a sandboxed so as to prevent the page from interfering with its
|
|
// content and behavior.
|
|
//
|
|
// The purpose of epicker.js is to:
|
|
// - Install the element picker UI, and wait for the component to establish
|
|
// a direct communication channel.
|
|
// - Lookup candidate filters from elements at a specific position.
|
|
// - Highlight element(s) at a specific position or according to whether
|
|
// they match candidate filters;
|
|
// - Preview the result of applying a candidate filter;
|
|
//
|
|
// When the element picker is installed on a page, the only change the page
|
|
// sees is an iframe with a random attribute. The page can't see the content
|
|
// of the iframe, and cannot interfere with its style properties. However the
|
|
// page can remove the iframe.
|
|
|
|
// We need extra messaging capabilities + fetch/process picker arguments.
|
|
{
|
|
const results = await Promise.all([
|
|
vAPI.messaging.extend(),
|
|
vAPI.messaging.send('elementPicker', { what: 'elementPickerArguments' }),
|
|
]);
|
|
if ( results[0] !== true ) { return; }
|
|
pickerBootArgs = results[1];
|
|
if ( typeof pickerBootArgs !== 'object' || pickerBootArgs === null ) {
|
|
return;
|
|
}
|
|
// Restore net filter union data if origin is the same.
|
|
const eprom = pickerBootArgs.eprom || null;
|
|
if ( eprom !== null && eprom.lastNetFilterSession === lastNetFilterSession ) {
|
|
lastNetFilterHostname = eprom.lastNetFilterHostname || '';
|
|
lastNetFilterUnion = eprom.lastNetFilterUnion || '';
|
|
}
|
|
}
|
|
|
|
// The DOM filterer will not be present when cosmetic filtering is disabled.
|
|
if (
|
|
pickerBootArgs.zap !== true &&
|
|
vAPI.domFilterer instanceof Object === false
|
|
) {
|
|
return;
|
|
}
|
|
|
|
// https://github.com/gorhill/uBlock/issues/1529
|
|
// In addition to inline styles, harden the element picker styles by using
|
|
// dedicated CSS rules.
|
|
const pickerCSSStyle = [
|
|
'background: transparent',
|
|
'border: 0',
|
|
'border-radius: 0',
|
|
'box-shadow: none',
|
|
'display: block',
|
|
'height: 100vh',
|
|
'left: 0',
|
|
'margin: 0',
|
|
'max-height: none',
|
|
'max-width: none',
|
|
'min-height: unset',
|
|
'min-width: unset',
|
|
'opacity: 1',
|
|
'outline: 0',
|
|
'padding: 0',
|
|
'pointer-events: auto',
|
|
'position: fixed',
|
|
'top: 0',
|
|
'visibility: visible',
|
|
'width: 100%',
|
|
'z-index: 2147483647',
|
|
''
|
|
].join(' !important;');
|
|
|
|
const pickerCSS = `
|
|
:root [${vAPI.sessionId}] {
|
|
${pickerCSSStyle}
|
|
}
|
|
:root [${vAPI.sessionId}-clickblind] {
|
|
pointer-events: none !important;
|
|
}
|
|
`;
|
|
|
|
vAPI.userStylesheet.add(pickerCSS);
|
|
vAPI.userStylesheet.apply();
|
|
|
|
pickerRoot = document.createElement('iframe');
|
|
pickerRoot.setAttribute(vAPI.sessionId, '');
|
|
document.documentElement.append(pickerRoot);
|
|
|
|
// https://github.com/gorhill/uBlock/issues/2060
|
|
if ( vAPI.domFilterer instanceof Object ) {
|
|
vAPI.domFilterer.excludeNode(pickerRoot);
|
|
}
|
|
|
|
vAPI.shutdown.add(quitPicker);
|
|
|
|
vAPI.MessagingConnection.addListener(onConnectionMessage);
|
|
|
|
{
|
|
const url = new URL(pickerBootArgs.pickerURL);
|
|
url.searchParams.set('epid', epickerId);
|
|
if ( pickerBootArgs.zap ) {
|
|
url.searchParams.set('zap', '1');
|
|
}
|
|
pickerRoot.contentWindow.location = url.href;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
|
|
})();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
DO NOT:
|
|
- Remove the following code
|
|
- Add code beyond the following code
|
|
Reason:
|
|
- https://github.com/gorhill/uBlock/pull/3721
|
|
- uBO never uses the return value from injected content scripts
|
|
|
|
**/
|
|
|
|
void 0;
|