Mirror of UserUtils' source code https://github.com/Sv443-Network/UserUtils
|
1 سال پیش | |
---|---|---|
.changeset | 1 سال پیش | |
.github | 1 سال پیش | |
lib | 1 سال پیش | |
.eslintrc.cjs | 1 سال پیش | |
.gitignore | 1 سال پیش | |
.npmignore | 1 سال پیش | |
CHANGELOG.md | 1 سال پیش | |
LICENSE.txt | 1 سال پیش | |
README.md | 1 سال پیش | |
package-lock.json | 1 سال پیش | |
package.json | 1 سال پیش | |
tsconfig.json | 1 سال پیش |
Library with various utilities for userscripts - register listeners for when CSS selectors exist, intercept events, modify the DOM more easily and more.
Contains builtin TypeScript declarations.
onSelector()
addEventListener()
on any given EventTarget objectaddEventListener()
on the window objectIf you are using a bundler like webpack, you can install this package using npm:
npm i @sv443-network/userutils
Then, import it in your script as usual:
import { addGlobalStyle } from "@sv443-network/userutils";
// or
import * as userUtils from "@sv443-network/userutils";
Shameless plug: I also have a webpack-based template for userscripts in TypeScript that you can use to get started quickly.
If you are not using a bundler, you can include the latest release from GreasyFork by adding this directive to the userscript header:
// @require https://greasyfork.org/scripts/TODO
If you like using this library, please consider supporting development
Usage:
onSelector<TElement = HTMLElement>(selector: string, options: {
listener: (elements: TElement | NodeListOf<TElement>) => void,
all?: boolean,
continuous?: boolean,
}): void
Registers a listener to be called whenever the element(s) behind a selector is/are found in the DOM.
If the selector already exists, the listener will be called immediately.
If all
is set to true
, querySelectorAll() will be used instead and the listener will return a NodeList of matching elements.
This will also include elements that were already found in a previous listener call.
If set to false
(default), querySelector() will be used and only the first matching element will be returned.
If continuous
is set to true
, the listener will not be deregistered after it was called once (defaults to false).
When using TypeScript, the generic TElement
can be used to specify the type of the element(s) that the listener will return.
⚠️ In order to use this function, initOnSelector()
has to be called as soon as possible.
This initialization function has to be called after DOMContentLoaded
is fired (or immediately if @run-at document-end
is set).
Calling onSelector() before DOMContentLoaded
is fired will not throw an error, but it also won't trigger listeners until the DOM is accessible.
Usage:
initOnSelector(options?: {
attributes?: boolean,
characterData?: boolean,
}): void
Initializes the MutationObserver that is used by onSelector()
to check for the registered selectors whenever a DOM change occurs on the <body>
By default, this only checks if elements are added or removed (at any depth).
Set attributes
to true
to also check for attribute changes on every single descendant of the <body>
(defaults to false).
Set characterData
to true
to also check for character data changes on every single descendant of the <body>
(defaults to false).
⚠️ Using these extra options can have a performance impact on larger sites or sites with a constantly changing DOM.
⚠️ This function needs to be run after the DOM has loaded (when using @run-at document-end
or after DOMContentLoaded
has fired).
Usage: getSelectorMap(): Map<string, OnSelectorOptions[]>
Returns a Map of all currently registered selectors and their options, including listener function.
Since multiple listeners can be registered for the same selector, the value of the Map is an array of OnSelectorOptions
objects.
Usage: autoPlural(str: string, num: number | Array | NodeList): string
Automatically pluralizes a string if the given number is not 1.
If an array or NodeList is passed, the length of it will be used.
Usage: clamp(num: number, min: number, max: number): number
Clamps a number between a min and max value.
Usage: pauseFor(ms: number): Promise<void>
Pauses async execution for a given amount of time.
Usage: debounce(func: Function, timeout?: number): Function
Debounces a function, meaning that it will only be called once after a given amount of time.
This is very useful for functions that are called repeatedly, like event listeners, to remove extraneous calls.
The timeout will default to 300ms if left undefined.
Usage: getUnsafeWindow(): Window
Returns the unsafeWindow object or falls back to the regular window object if the @grant unsafeWindow
is not given.
Userscripts are sandboxed and do not have access to the regular window object, so this function is useful for websites that reject some events that were dispatched by the userscript.
Usage: insertAfter(beforeElement: HTMLElement, afterElement: HTMLElement): HTMLElement
Inserts the element passed as afterElement
as a sibling after the passed beforeElement
.
The passed afterElement
will be returned.
⚠️ This function needs to be run after the DOM has loaded (when using @run-at document-end
or after DOMContentLoaded
has fired).
Usage: addParent(element: HTMLElement, newParent: HTMLElement): HTMLElement
Adds a parent element around the passed element
and returns the new parent.
Previously registered event listeners are kept intact.
⚠️ This function needs to be run after the DOM has loaded (when using @run-at document-end
or after DOMContentLoaded
has fired).
Usage: addGlobalStyle(css: string): void
Adds a global style to the page in form of a <style>
element that's inserted into the <head>
.
⚠️ This function needs to be run after the DOM has loaded (when using @run-at document-end
or after DOMContentLoaded
has fired).
Usage: preloadImages(urls: string[], rejects?: boolean): Promise<void>
Preloads images into browser cache by creating an invisible <img>
element for each URL passed.
The images will be loaded in parallel and the returned Promise will only resolve once all images have been loaded.
The resulting PromiseSettledResult array will contain the image elements if resolved, or an ErrorEvent if rejected, but only if rejects
is set to true.
Usage:
fetchAdvanced(url: string, options?: {
timeout?: number,
// any other options from fetch() except for signal
}): Promise<Response>
A wrapper around the native fetch()
function that adds options like a timeout property.
The timeout will default to 10 seconds if left undefined.
Usage: openInNewTab(url: string): void
Creates an invisible anchor with a _blank
target and clicks it.
Contrary to window.open()
, this has a lesser chance to get blocked by the browser's popup blocker and doesn't open the URL as a new window.
This function has to be run in relatively quick succession in response to a user interaction event, else the browser might reject it.
⚠️ This function needs to be run after the DOM has loaded (when using @run-at document-end
or after DOMContentLoaded
has fired).
Usage: interceptEvent(eventObject: EventTarget, eventName: string, predicate: () => boolean): void
Intercepts all events dispatched on the eventObject
and prevents the listeners from being called as long as the predicate function returns a truthy value.
Calling this function will set the Error.stackTraceLimit
to 1000 (if it's not already higher) to ensure the stack trace is preserved.
⚠️ This function should be called as soon as possible (I recommend using @run-at document-start
), as it will only intercept events that are attached after this function is called.
Usage: interceptWindowEvent(eventName: string, predicate: () => boolean): void
Intercepts all events dispatched on the window
object and prevents the listeners from being called as long as the predicate function returns a truthy value.
This is essentially the same as interceptEvent()
, but automatically uses the unsafeWindow
(or falls back to regular window
).
⚠️ This function should be called as soon as possible (I recommend using @run-at document-start
), as it will only intercept events that are attached after this function is called.
This library is licensed under the MIT License.
See the license file for details.