Use as an npm package:
npm i @webqit/dom
// Import
import init from '@webqit/dom';
// Initialize the lib
init.call( window );
// Obtain the APIs
const { ready, Realtime, Reflow } = window.webqit.dom;
Use as a script:
<script src="https://unpkg.com/@webqit/dom/dist/main.js"></script>
// Obtain the APIs
const { ready, Realtime, Reflow } = window.webqit.dom;
Know when the document is ready.
ready(() => {
console.log( 'Document is ready' );
});
React to realtime DOM operations.
Realtime.attr( context, callback )
Realtime.attr( context, filter, callback )
A succinct attributes observer API that abstracts the MutationObserver API!
// Observe all attributes that have been added or removed to/from the specified context ("div" in this case)
Realtime.attr( div, logMutationRecord );
// Observe when the specified attributes are added or removed to/from the specified context ("div" in this case)
Realtime.attr( div, [ 'contenteditable', 'data-state' ], logMutationRecord );
function logMutationRecord( record, context ) {
// Note the record.name and record.oldValue properties
console.log( record.target, record.name, record.oldValue, record.type === 'attribute-record' );
}
Realtime.observe( context, callback[, params = {} ])
Realtime.observe( context, filter, callback[, params = {} ])
A beautiful abstraction over the awful MutationObserver API!
// Observe all elements that have been added or removed to/from the specified context (document in this case)
Realtime.observe( document, logMutationRecord, { subtree: true } );
// Observe all "p" elements that have been added or removed to/from the specified context (document in this case)
Realtime.observe( document, 'p', logMutationRecord, { subtree: true } );
// "p" elements - whether added from markup and deeply nested (as per { subtree: true })...
document.body.innerHTML = '<div><p></p></div>';
// or added programmatically...
const p = document.createElement( 'p' );
const div = document.createElement( 'div' );
// and deeply nested (as per { subtree: true })
div.appendChild( p );
document.body.appendChild( div );
// Observe element instances too (e.g. a "p" instance)...
const p = document.createElement( 'p' );
Realtime.observe( document, [ p, orCssSelector ], logMutationRecord, { subtree: true } );
// for when they've been added or removed to/from the given context (document in this case)...
const div = document.createElement( 'div' );
// and even when deeply nested (as per { subtree: true })
div.appendChild( p );
document.body.appendChild( div );
// And now, when having been removed from context...
// either via an overwrite... (indirect overwrite in this case, as per { subtree: true })...
document.body.innerHTML = '';
// or via some programmatic means... (indirect removal in this case, as per { subtree: true })...
document.querySelector( 'div' ).remove();
function logMutationRecord( record, context ) {
// Note the record.addedNodes and record.removedNodes arrays
console.log( record.target, record.addedNodes, record.removedNodes, record.type === 'mutation-record' );
}
Realtime.match( context, callback[, params = {} ])
Realtime.match( context, filter, callback[, params = {} ])
A dual-purpose method that both delivers the current matching result and keeps it live by employing Realtime.observe()
under the hood.
// This becomes an exact alias for Realtime.observe() being that no targets are specified
Realtime.match( document, logMutationRecord, { subtree: true } );
// Now, deliver all current "p" elements and keep subsequent mutations to "p" elements coming
Realtime.match( document, 'p', logMutationRecord, { subtree: true } );
// Match element instances too (e.g. a "p" instance)...
const p = document.createElement( 'p' );
Realtime.match( document, [ p, orCssSelector ], logMutationRecord, { subtree: true } );
// But "p" doesn't match as a node connected to the context (document in this case)
// and so isn't delivered
// But it's caught by the observer when added to the context
const div = document.createElement( 'div' );
div.appendChild( p );
document.body.appendChild( div );
function logMutationRecord( record, context ) {
// Depending on record.type
// Note the record.connectedNodes and record.disconnectedNodes arrays
console.log( record.target, record.connectedNodes, record.disconnectedNodes, record.type === 'query-record' );
// Note the record.addedNodes and record.removedNodes arrays
console.log( record.target, record.addedNodes, record.removedNodes, record.type === 'mutation-record' );
}
Realtime.intercept( context, callback[, params = {} ])
Realtime.intercept( context, filter, callback[, params = {} ])
An ahead-of-time mutation observer API that intercepts DOM operations before they happen. This is much like Realtime.observe()
but with a marked difference: timing! This captures mutations that are about to happen, while the former captures mutations that have just happened!
A good usecase is ahead.
// Intercept all elements that are BEING added or removed to/from the specified context (document in this case)
Realtime.intercept( document, logInterceptionRecord, { subtree: true } );
// Intercept all "p" elements that are BEING added or removed to/from the specified context (document in this case)
Realtime.intercept( document, 'p', logInterceptionRecord, { subtree: true } );
// "p" elements - whether added from markup and deeply nested (as per { subtree: true })...
document.body.innerHTML = '<div><p></p></div>';
// or added programmatically...
const p = document.createElement( 'p' );
const div = document.createElement( 'div' );
// and deeply nested (as per { subtree: true })
div.appendChild( p );
document.body.appendChild( div );
// Intercept element instances too (e.g. a "p" instance)...
const p = document.createElement( 'p' );
Realtime.intercept( document, [ p, orCssSelector ], logInterceptionRecord, { subtree: true } );
// for when they're BEING added or removed to/from the given context (document in this case)...
const div = document.createElement( 'div' );
// and even when deeply nested (as per { subtree: true })
div.appendChild( p );
document.body.appendChild( div );
// And now, when BEING removed from context...
// either via an overwrite... (indirect overwrite in this case, as per { subtree: true })...
document.body.innerHTML = '';
// or via some programmatic means... (indirect removal in this case, as per { subtree: true })...
document.querySelector( 'div' ).remove();
function logInterceptionRecord( record, context ) {
// Note the record.incomingNodes and record.outgoingNodes arrays
console.log( record.target, record.incomingNodes, record.outgoingNodes, record.type === 'interception-record' );
}
Some niceties
-
With each of the three APIs, it is possible to opt in to either just the "connected", "added", "incoming" records or to just the "disconnected", "removed", "outgoing" records. You'd use the
params.on
property:params.on: 'connected'
- only records for "connected", "added", "incoming" nodes are delivered - with thematch()
,observe()
,intercept()
APIs respectively.params.on: 'disconnected'
- only records for "disconnected", "removed", "outgoing" nodes are delivered - with thematch()
,observe()
,intercept()
APIs respectively.
-
With each of the three APIs, omiting the
{ subtree: true }
setting would mean that deeply nested targets won't be searched for; only directly-mutated nodes will be evaluated. -
The
Realtime
API is designed for the consistency and predictability that the nativeMutationObserver
API lacks for certain usecases.For example, bind a mutation observer - with
{subtree: true}
- to thedocument
object before page parsing begins, and you'd see that all elements are announced:<html> <head> <script> new MutationObserver( records => { // Log mutations and notice that every element in the tree - e.g. <div> and <p> - is caught } ).observe( document, { subtree: true } ); </script> </head> <body> <div> <p></p> </div> </body> </html>
But try adding an equivalent DOM structure programmatically - e.g.
<div><p></p></div>
- and you'd see that nested elements (p
) aren't caught:const div = document.createElement( 'div' ); const p = document.createElement( 'p' ); div.appendChild( p ); document.body.appendChild( div );
By contrast, the
Realtime
API is consistent with{ subtree: true }
in all cases! -
The
Realtime.intercept()
API is designed for the rare possiblity of intercepting elements before they're handled natively by the browser. This lets you build tools that extend the DOM in more low-level ways. For example, you could intercept and rewrite<script>
elements before they're parsed and executed.
Some notes
-
The
Realtime
API is able to do the extra-ordinary by going a bit extra-ordinary: by literally intercepting DOM APIs. And here is the complete list of them:Node
:insertBefore
,replaceChild
,removeChild
,appendChild
,textContent
,nodeValue
.Element
:insertAdjacentElement
,insertAdjacentHTML
,setHTML
,replaceChildren
,replaceWith
,remove
.before
,after
,append
,prepend
.HTMLElement
:outerText
,innerText
.
Point is: monkeying (responsibly) with the DOM for polyfill development is a norm. But you may need to consider this caveat carefully in your specific usecases.
Eliminate layout thrashing by batching DOM read/write operations. (Compare fastdom)
Reflow.onread( () => {
console.log( 'reading phase of the UI' );
} );
Reflow.onwrite( () => {
console.log( 'writing phase of the UI' );
} );
Reflow.onread( () => {
console.log( 'reading phase of the UI' );
} );
Reflow.onwrite( () => {
console.log( 'writing phase of the UI' );
} );
reading phase of the UI
reading phase of the UI
writing phase of the UI
writing phase of the UI
Concept
The Reflow
API works as a regulatory layer between your app/library and the DOM. It lets you think of the DOM in terms of a "reading" phase and a "writing" phase, and lets you hook into this cycle when working with the DOM: onread()
for doing "read" operations, and onwrite
for doing "write" operations. Batching DOM operations this way lets us avoid unnecessary document reflows and dramatically speed up layout performance.
Each read/write operation is added to a corresponding read/write queue. The queues are emptied (reads, then writes) at the turn of the next frame using
window.requestAnimationFrame
.
Reflow.onread( onread[, inPromiseMode = false ])
Schedules a job for the "read" phase. Can return a promise that resolves when job eventually executes; you ask for a promise by supplying true
as a second argument.
const promise = Reflow .onread( () => {
const width = element.clientWidth;
}, true/*give back a promise*/ );
Reflow.onwrite( onwrite[, inPromiseMode = false ])
Schedules a job for the "write" phase. Can return a promise that resolves when job eventually executes; you ask for a promise by supplying true
as a second argument.
const promise = Reflow .onwrite( () => {
element.style.width = width + 'px';
}, true/*give back a promise*/ );
Reflow.cycle( onread, onwrite )
Puts your read/write operations in a cycle that keeps in sync with the UI's read/write phases.
Reflow.cycle(
// onread
() => {
// Do a read operation
const width = element.clientWidth;
// Now if we return anything other than undefined, the "onwrite" block is executed
return width; // recieved by the "onwrite" callback on its first parameter
},
// onwrite
( width, carried ) => {
// Do a write operation
element.style.width = width + 'px';
// Now if we return anything other than undefined, the cycle repeats starting with the "onread" block
return newCarry; // recieved by the "onwrite" block again on its second parameter: "carried"
}
);
To report bugs or request features, please submit an issue.
MIT.