Security News
PyPI Introduces Digital Attestations to Strengthen Python Package Security
PyPI now supports digital attestations, enhancing security and trust by allowing package maintainers to verify the authenticity of Python packages.
@broadly/wicg-inert
Advanced tools
inert
attributeTable of Contents
inert
as a primitive
The inert
attribute would allow web authors to mark parts of the DOM tree as inert:
When a node is inert, then the user agent must act as if the node was absent for the purposes of targeting user interaction events, may ignore the node for the purposes of text search user interfaces (commonly known as "find in page"), and may prevent the user from selecting text in that node.
Furthermore, a node which is inert should also be hidden from assistive technology.
Try out the polyfill, or look at the demo page.
The inert
attribute was originally specced as part of the <dialog>
element specification.
<dialog>
required the concept of inert
to be defined in order to describe the blocking behaviour of dialogs,
and the inert
attribute was introduced "so you could do <dialog>
without <dialog>
".
The attribute was later removed as it was argued that its only use case was subsumed by <dialog>
. However, later discussion on the original bug proposed several use cases which could not be handled, or only handled poorly, using <dialog>
.
The spec for the inert
attribute,
with the existing definition of "inert" already specified,
is extremely straightforward:
The
inert
attribute
The
inert
attribute is a boolean attribute that indicates, by its presence, that the element is to be made inert.
When an element has an
inert
attribute, the user agent must mark that element as inert.
By default, there is no visual indication of a subtree being inert. Authors are encouraged to clearly mark what parts of their document are active and which are inert, to avoid user confusion. In particular, it is worth remembering that not all users can see all parts of a page at once; for example, users of screen readers, users on small devices or with magnifiers, and even users just using particularly small windows might not be able to see the active part of a page and may get frustrated if inert sections are not obviously inert. For individual controls, the
disabled
attribute is probably more appropriate.
The spec does not explicitly state what effect inert
has on the subtree of the element marked as inert
,
however it is implied by the note that inert
causes the entire subtree of the element with the inert
attribute to be made inert.
The polyfill makes the assumption that the entire subtree becomes inert.
aria-hidden
and lang
imply that it should.
The polyfill assumes that it does so.The existing description of inert is not specific about where pointer events which would have been targeted to an element in an inert subtree should go. (See also: discussion on the WHATWG pull request.) Does the event:
Consistency with pointer-events
would suggest (ii). The polyfill uses pointer-events: none
and so models its behaviour.
The spec is also not explicit about whether the attribute should be reflected. The polyfill assumes that it is.
The spec does not explicitly state that inert content should be hidden from assistive technology.
However, typically, the HTML spec does not provide this type of information. The polyfill makes inert content hidden from assistive technology (via aria-hidden
).
The spec does not make explicit that there is no way to "un-inert" a subtree of an inert subtree.
inert
as a primitiveDevelopers find themselves in situations where they'd like to be able to mark a part of the page "un-tabbable". Rob Dodson lays out one such example in his article "Building better accessibility primitives":
One problem: to [achieve a performance optimisation for animation] we must leave the drawer in the DOM at all times. Meaning its focusable children are just sitting there offscreen, and as the user is tabbing through the page eventually their focus will just disappear into the drawer and they won't know where it went. I see this on responsive websites all the time. This is just one example but I've also run into the need to disable tabindex when I'm animating between elements with opacity: 0, or temporarily disabling large lists of custom controls, and as others have pointed out, you'd hit if you tried to build something like coverflow where you can see a preview of the next element but can't actually interact with it yet.
inert
would also allow slightly more straightforward polyfilling of both <dialog>
and the proposed, more primitive
blockingElements
API.
See
Polymer Labs' blockingElements
polyfill,
based on this polyfill,
for an example of how inert
may be used for this purpose.
Currently, since there is no way to express the "inertness" concept,
polyfilling these APIs requires both focus event trapping
to avoid focus cycling out of the dialog/blocking element
(and thus as a side effect may prevent focus from walking out of the page at all)
and a tree-walk
(usually neglected by developers)
to set aria-hidden
on all sibling elements of the dialog or blocking element.
On the implementer side,
the vast majority of work involved in implementing inert
is a necessary pre-cursor to both <dialog>
and blockingElements
implementations,
so by implementing inert
first,
implementers may get useful functionality into the hands of developers sooner while still laying the groundwork for one or both of these more complex APIs.
Temporarily offscreen/hidden content
As discussed in the article, there are a range of circumstances in which case it's desirable to add content to the DOM to be rendered but remain offscreen.
In these cases, without inert
, authors are forced to choose between
an accessible experience for keyboard and assistive technology users,
or the factors (such as performance) which make offscreen rendering desirable -
or, performing all the contortions necessary to keep the offscreen content functionally "inert".
These cases include:
opacity
to zero,
and animates transitions between items;On-screen but non-interactive content
Occasionally, UI designs require that certain content be visible or partially visible,
but clearly non-interactive.
Typically, this content is made non-interactive for pointer device users
either via a semi-transparent overlay which provides a visual cue as well as intercepting pointer events,
or via using pointer-events: none
.
In these cases developers are once again required to perform contortions in order to ensure that this content is not an accessibility issue.
These cases include:
Any of the use cases for blockingElement[s]
:
A slide show or "cover flow" style carousel may have non-active items partially visible, as a preview - they may be transformed or partially obscured to indicate that they are non-interactive.
Form content which is not currently relevant, e.g. fading out and disabling the "Shipping Address" fields when the "Same as billing address" checkbox has been checked.
Disabling the entire UI while in an inconsistent state, such as showing a throbber/loading bar during unexpectedly slow loading.
A CSS property?
inert
encompasses the behaviour of at least two other things which are CSS properties -
pointer-events: none
and user-select: none
, plus another attribute, aria-hidden
.
These behaviours, along with the currently near-impossible to achieve behaviour of preventing tabbing/programmatic focus, are very frequently applied together
(or if one, such as aria-hidden
, is omitted, it is more often through lack of awareness than deliberate).
There is scope for a more primitive CSS property to "explain" the ability of inert
to prevent focus, however that could easily coexist with the inert
attribute.
blockingElements
(or, potentially, a single blockingElement
) represents roughly the opposite use case to inert
:
a per-document, single element which blocks the document, analogous to the blocking behaviour of a modal dialog.
It's not always the case that we will want a single subtree to be non-inert. Ideally, we would have both concepts available;
however, inert
allows reasonable approximation of blockingElements
whereas the reverse is not true.
blockingElement
using inert
, it's most straightforward to insert a non-inert element as a sibling element to the main page content, and then use inert
to mark the main page content as inert.
More generally, all siblings of the desired "blocking" element, plus all siblings of all of its ancestors, could be marked inert.A programmatic API?
Something like document.makeInert(el)
.
This would require waiting for script execution before parts of the page became inert, which can take some time.
The dist
directory contains a transpiled, UMD build of inert.js
generated from the ES6 source.
There is also an HTML Import
available in the root of the project at inert.html
.
The polyfill attempts to provide a reasonable fidelity polyfill for the inert
attribute, however please note:
It relies on mutation observers to detect the addition of the inert
attribute, and to detect dynamically added content within inert subtrees.
Testing for inert-ness in any way immediately after either type of mutation will therefore give inconsistent results;
please allow the current task to end before relying on mutation-related changes to take effect, for example via Promise.resolve()
.
Example:
const newButton = document.createElement('button');
const inertContainer = document.querySelector('[inert]');
inertContainer.appendChild(newButton);
// Wait for the next microtask to allow mutation observers to react to the DOM change
Promise.resolve().then(() => {
expect(isUnfocusable(newButton)).to.equal(true);
});
Using the inert
property, however, is synchronous.
It will be very expensive performance-wise compared to a native inert
implementation, because it requires a lot of tree-walking on any relevant mutation
(applying inert
, or adding descendant content into inert
subtrees).
npm install --save wicg-inert
If you want to use inert
with an older browser or JavaScript runtime like PhantomJS you'll need to include a polyfill for Map and Set, such as babel-polyfill.
As suggested on StackOverflow, try adding the following to the document in your build process:
<script src="https://cdnjs.cloudflare.com/ajax/libs/babel-polyfill/6.16.0/polyfill.min.js"></script>
FAQs
A polyfill for the proposed inert API
The npm package @broadly/wicg-inert receives a total of 3 weekly downloads. As such, @broadly/wicg-inert popularity was classified as not popular.
We found that @broadly/wicg-inert demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
PyPI now supports digital attestations, enhancing security and trust by allowing package maintainers to verify the authenticity of Python packages.
Security News
GitHub removed 27 malicious pull requests attempting to inject harmful code across multiple open source repositories, in another round of low-effort attacks.
Security News
RubyGems.org has added a new "maintainer" role that allows for publishing new versions of gems. This new permission type is aimed at improving security for gem owners and the service overall.