2022-09-13 23:44:24 +02:00
|
|
|
/*******************************************************************************
|
|
|
|
|
2023-12-04 18:10:34 +01:00
|
|
|
uBlock Origin - a comprehensive, efficient content blocker
|
2022-09-13 23:44:24 +02:00
|
|
|
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
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* jshint esversion:11 */
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2022-09-15 19:14:08 +02:00
|
|
|
const normalizeTarget = target => {
|
2022-11-12 15:51:22 +01:00
|
|
|
if ( typeof target === 'string' ) { return Array.from(qsa$(target)); }
|
|
|
|
if ( target instanceof Element ) { return [ target ]; }
|
2022-09-13 23:44:24 +02:00
|
|
|
if ( target === null ) { return []; }
|
2022-11-12 15:51:22 +01:00
|
|
|
if ( Array.isArray(target) ) { return target; }
|
|
|
|
return Array.from(target);
|
2022-09-15 19:14:08 +02:00
|
|
|
};
|
2022-09-13 23:44:24 +02:00
|
|
|
|
2022-09-15 19:14:08 +02:00
|
|
|
const makeEventHandler = (selector, callback) => {
|
2022-09-13 23:44:24 +02:00
|
|
|
return function(event) {
|
|
|
|
const dispatcher = event.currentTarget;
|
|
|
|
if (
|
|
|
|
dispatcher instanceof HTMLElement === false ||
|
|
|
|
typeof dispatcher.querySelectorAll !== 'function'
|
|
|
|
) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const receiver = event.target;
|
|
|
|
const ancestor = receiver.closest(selector);
|
|
|
|
if (
|
|
|
|
ancestor === receiver &&
|
|
|
|
ancestor !== dispatcher &&
|
|
|
|
dispatcher.contains(ancestor)
|
|
|
|
) {
|
|
|
|
callback.call(receiver, event);
|
|
|
|
}
|
|
|
|
};
|
2022-09-15 19:14:08 +02:00
|
|
|
};
|
2022-09-13 23:44:24 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
|
|
|
class dom {
|
|
|
|
static attr(target, attr, value = undefined) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
if ( value === undefined ) {
|
|
|
|
return elem.getAttribute(attr);
|
|
|
|
}
|
2022-09-30 01:51:33 +02:00
|
|
|
if ( value === null ) {
|
|
|
|
elem.removeAttribute(attr);
|
|
|
|
} else {
|
|
|
|
elem.setAttribute(attr, value);
|
|
|
|
}
|
2022-09-13 23:44:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-06 18:50:25 +02:00
|
|
|
static clear(target) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
while ( elem.firstChild !== null ) {
|
|
|
|
elem.removeChild(elem.firstChild);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-12 15:51:22 +01:00
|
|
|
static clone(target) {
|
2023-05-06 18:50:25 +02:00
|
|
|
const elements = normalizeTarget(target);
|
|
|
|
if ( elements.length === 0 ) { return null; }
|
|
|
|
return elements[0].cloneNode(true);
|
2022-11-12 15:51:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static create(a) {
|
|
|
|
if ( typeof a === 'string' ) {
|
|
|
|
return document.createElement(a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-06 18:50:25 +02:00
|
|
|
static prop(target, prop, value = undefined) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
if ( value === undefined ) { return elem[prop]; }
|
|
|
|
elem[prop] = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-15 19:14:08 +02:00
|
|
|
static text(target, text) {
|
2022-11-12 16:35:40 +01:00
|
|
|
const targets = normalizeTarget(target);
|
|
|
|
if ( text === undefined ) {
|
|
|
|
return targets.length !== 0 ? targets[0].textContent : undefined;
|
|
|
|
}
|
|
|
|
for ( const elem of targets ) {
|
2022-09-15 19:14:08 +02:00
|
|
|
elem.textContent = text;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-13 23:44:24 +02:00
|
|
|
static remove(target) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
elem.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-12 15:51:22 +01:00
|
|
|
// target, type, callback, [options]
|
|
|
|
// target, type, subtarget, callback, [options]
|
|
|
|
|
|
|
|
static on(target, type, subtarget, callback, options) {
|
|
|
|
if ( typeof subtarget === 'function' ) {
|
|
|
|
options = callback;
|
|
|
|
callback = subtarget;
|
|
|
|
subtarget = undefined;
|
|
|
|
if ( typeof options === 'boolean' ) {
|
|
|
|
options = { capture: true };
|
|
|
|
}
|
2022-09-13 23:44:24 +02:00
|
|
|
} else {
|
2022-11-12 15:51:22 +01:00
|
|
|
callback = makeEventHandler(subtarget, callback);
|
|
|
|
if ( options === undefined || typeof options === 'boolean' ) {
|
|
|
|
options = { capture: true };
|
|
|
|
} else {
|
|
|
|
options.capture = true;
|
|
|
|
}
|
2022-09-13 23:44:24 +02:00
|
|
|
}
|
2022-11-12 15:51:22 +01:00
|
|
|
const targets = target instanceof Window || target instanceof Document
|
|
|
|
? [ target ]
|
|
|
|
: normalizeTarget(target);
|
|
|
|
for ( const elem of targets ) {
|
|
|
|
elem.addEventListener(type, callback, options);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static off(target, type, callback, options) {
|
|
|
|
if ( typeof callback !== 'function' ) { return; }
|
|
|
|
if ( typeof options === 'boolean' ) {
|
|
|
|
options = { capture: true };
|
|
|
|
}
|
|
|
|
const targets = target instanceof Window || target instanceof Document
|
|
|
|
? [ target ]
|
|
|
|
: normalizeTarget(target);
|
|
|
|
for ( const elem of targets ) {
|
|
|
|
elem.removeEventListener(type, callback, options);
|
2022-09-13 23:44:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-15 19:14:08 +02:00
|
|
|
dom.cl = class {
|
|
|
|
static add(target, name) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
elem.classList.add(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static remove(target, name) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
elem.classList.remove(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static toggle(target, name, state) {
|
2022-11-12 15:51:22 +01:00
|
|
|
let r;
|
2022-09-15 19:14:08 +02:00
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
2022-11-12 15:51:22 +01:00
|
|
|
r = elem.classList.toggle(name, state);
|
2022-09-15 19:14:08 +02:00
|
|
|
}
|
2022-11-12 15:51:22 +01:00
|
|
|
return r;
|
2022-09-15 19:14:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static has(target, name) {
|
|
|
|
for ( const elem of normalizeTarget(target) ) {
|
|
|
|
if ( elem.classList.contains(name) ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-09-13 23:44:24 +02:00
|
|
|
/******************************************************************************/
|
|
|
|
|
2022-11-12 15:51:22 +01:00
|
|
|
function qs$(a, b) {
|
|
|
|
if ( typeof a === 'string') {
|
|
|
|
return document.querySelector(a);
|
|
|
|
}
|
2023-05-06 18:50:25 +02:00
|
|
|
if ( a === null ) { return null; }
|
2022-11-12 15:51:22 +01:00
|
|
|
return a.querySelector(b);
|
2022-09-13 23:44:24 +02:00
|
|
|
}
|
|
|
|
|
2022-11-12 15:51:22 +01:00
|
|
|
function qsa$(a, b) {
|
|
|
|
if ( typeof a === 'string') {
|
|
|
|
return document.querySelectorAll(a);
|
|
|
|
}
|
2023-05-06 18:50:25 +02:00
|
|
|
if ( a === null ) { return []; }
|
2022-11-12 15:51:22 +01:00
|
|
|
return a.querySelectorAll(b);
|
2022-09-13 23:44:24 +02:00
|
|
|
}
|
|
|
|
|
2022-11-12 15:51:22 +01:00
|
|
|
dom.root = qs$(':root');
|
|
|
|
dom.html = document.documentElement;
|
|
|
|
dom.head = document.head;
|
|
|
|
dom.body = document.body;
|
2022-09-15 19:14:08 +02:00
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
|
2022-09-13 23:44:24 +02:00
|
|
|
export { dom, qs$, qsa$ };
|